Adding quic geyser source and making it work with all raydium pools (#5)
Adding quic geyser source and making it work with all raydium pools
This commit is contained in:
parent
31b4f35d61
commit
520d6a74b3
|
@ -740,6 +740,8 @@ dependencies = [
|
||||||
"ordered-float",
|
"ordered-float",
|
||||||
"priority-queue",
|
"priority-queue",
|
||||||
"prometheus",
|
"prometheus",
|
||||||
|
"quic-geyser-client",
|
||||||
|
"quic-geyser-common",
|
||||||
"rand 0.7.3",
|
"rand 0.7.3",
|
||||||
"regex",
|
"regex",
|
||||||
"router-config-lib",
|
"router-config-lib",
|
||||||
|
@ -3264,7 +3266,7 @@ dependencies = [
|
||||||
"futures-util",
|
"futures-util",
|
||||||
"http 0.2.12",
|
"http 0.2.12",
|
||||||
"hyper 0.14.28",
|
"hyper 0.14.28",
|
||||||
"rustls 0.21.10",
|
"rustls 0.21.12",
|
||||||
"tokio",
|
"tokio",
|
||||||
"tokio-rustls 0.24.1",
|
"tokio-rustls 0.24.1",
|
||||||
]
|
]
|
||||||
|
@ -5303,6 +5305,40 @@ dependencies = [
|
||||||
"syn 2.0.58",
|
"syn 2.0.58",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "quic-geyser-client"
|
||||||
|
version = "0.1.5"
|
||||||
|
source = "git+https://github.com/blockworks-foundation/quic_geyser_plugin.git?branch=router_v1.17.29#8efcc200c795b1236675b161c04e5e65e00ace48"
|
||||||
|
dependencies = [
|
||||||
|
"anyhow",
|
||||||
|
"bincode",
|
||||||
|
"log 0.4.21",
|
||||||
|
"pkcs8",
|
||||||
|
"quic-geyser-common",
|
||||||
|
"quinn",
|
||||||
|
"rcgen",
|
||||||
|
"rustls 0.21.12",
|
||||||
|
"solana-sdk",
|
||||||
|
"tokio",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "quic-geyser-common"
|
||||||
|
version = "0.1.5"
|
||||||
|
source = "git+https://github.com/blockworks-foundation/quic_geyser_plugin.git?branch=router_v1.17.29#8efcc200c795b1236675b161c04e5e65e00ace48"
|
||||||
|
dependencies = [
|
||||||
|
"anyhow",
|
||||||
|
"bincode",
|
||||||
|
"itertools 0.10.5",
|
||||||
|
"log 0.4.21",
|
||||||
|
"lz4",
|
||||||
|
"serde",
|
||||||
|
"solana-program",
|
||||||
|
"solana-sdk",
|
||||||
|
"solana-transaction-status",
|
||||||
|
"thiserror",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "quick-protobuf"
|
name = "quick-protobuf"
|
||||||
version = "0.8.0"
|
version = "0.8.0"
|
||||||
|
@ -5323,7 +5359,7 @@ dependencies = [
|
||||||
"quinn-proto",
|
"quinn-proto",
|
||||||
"quinn-udp",
|
"quinn-udp",
|
||||||
"rustc-hash",
|
"rustc-hash",
|
||||||
"rustls 0.21.10",
|
"rustls 0.21.12",
|
||||||
"thiserror",
|
"thiserror",
|
||||||
"tokio",
|
"tokio",
|
||||||
"tracing",
|
"tracing",
|
||||||
|
@ -5339,7 +5375,7 @@ dependencies = [
|
||||||
"rand 0.8.5",
|
"rand 0.8.5",
|
||||||
"ring 0.16.20",
|
"ring 0.16.20",
|
||||||
"rustc-hash",
|
"rustc-hash",
|
||||||
"rustls 0.21.10",
|
"rustls 0.21.12",
|
||||||
"rustls-native-certs",
|
"rustls-native-certs",
|
||||||
"slab",
|
"slab",
|
||||||
"thiserror",
|
"thiserror",
|
||||||
|
@ -5762,7 +5798,7 @@ dependencies = [
|
||||||
"once_cell",
|
"once_cell",
|
||||||
"percent-encoding 2.3.1",
|
"percent-encoding 2.3.1",
|
||||||
"pin-project-lite",
|
"pin-project-lite",
|
||||||
"rustls 0.21.10",
|
"rustls 0.21.12",
|
||||||
"rustls-pemfile",
|
"rustls-pemfile",
|
||||||
"serde",
|
"serde",
|
||||||
"serde_json",
|
"serde_json",
|
||||||
|
@ -6049,9 +6085,9 @@ dependencies = [
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "rustls"
|
name = "rustls"
|
||||||
version = "0.21.10"
|
version = "0.21.12"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "f9d5a6813c0759e4609cd494e8e725babae6a2ca7b62a5536a13daaec6fcb7ba"
|
checksum = "3f56a14d1f48b391359b22f731fd4bd7e43c97f3c50eee276f3aa09c94784d3e"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"log 0.4.21",
|
"log 0.4.21",
|
||||||
"ring 0.17.8",
|
"ring 0.17.8",
|
||||||
|
@ -6117,8 +6153,8 @@ dependencies = [
|
||||||
"sanctum-token-ratio",
|
"sanctum-token-ratio",
|
||||||
"solana-program",
|
"solana-program",
|
||||||
"solana-readonly-account",
|
"solana-readonly-account",
|
||||||
"spl-associated-token-account 2.3.0",
|
"spl-associated-token-account 1.1.3",
|
||||||
"spl-token 4.0.0",
|
"spl-token 3.5.0",
|
||||||
"spl-token-metadata-interface",
|
"spl-token-metadata-interface",
|
||||||
"static_assertions",
|
"static_assertions",
|
||||||
]
|
]
|
||||||
|
@ -6231,7 +6267,7 @@ source = "git+https://github.com/igneous-labs/sanctum-solana-utils.git?rev=2d171
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"solana-program",
|
"solana-program",
|
||||||
"solana-readonly-account",
|
"solana-readonly-account",
|
||||||
"spl-associated-token-account 2.3.0",
|
"spl-associated-token-account 1.1.3",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
|
@ -6284,7 +6320,7 @@ source = "git+https://github.com/igneous-labs/sanctum-solana-utils.git?rev=2d171
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"solana-program",
|
"solana-program",
|
||||||
"solana-readonly-account",
|
"solana-readonly-account",
|
||||||
"spl-token-2022 1.0.0",
|
"spl-token-2022 0.6.1",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
|
@ -7408,7 +7444,7 @@ dependencies = [
|
||||||
"quinn",
|
"quinn",
|
||||||
"quinn-proto",
|
"quinn-proto",
|
||||||
"rcgen",
|
"rcgen",
|
||||||
"rustls 0.21.10",
|
"rustls 0.21.12",
|
||||||
"solana-connection-cache",
|
"solana-connection-cache",
|
||||||
"solana-measure",
|
"solana-measure",
|
||||||
"solana-metrics",
|
"solana-metrics",
|
||||||
|
@ -7725,7 +7761,7 @@ dependencies = [
|
||||||
"quinn-proto",
|
"quinn-proto",
|
||||||
"rand 0.8.5",
|
"rand 0.8.5",
|
||||||
"rcgen",
|
"rcgen",
|
||||||
"rustls 0.21.10",
|
"rustls 0.21.12",
|
||||||
"solana-metrics",
|
"solana-metrics",
|
||||||
"solana-perf",
|
"solana-perf",
|
||||||
"solana-sdk",
|
"solana-sdk",
|
||||||
|
@ -9084,7 +9120,7 @@ version = "0.24.1"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "c28327cf380ac148141087fbfb9de9d7bd4e84ab5d2c28fbc911d753de8a7081"
|
checksum = "c28327cf380ac148141087fbfb9de9d7bd4e84ab5d2c28fbc911d753de8a7081"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"rustls 0.21.10",
|
"rustls 0.21.12",
|
||||||
"tokio",
|
"tokio",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
@ -9158,7 +9194,7 @@ checksum = "212d5dcb2a1ce06d81107c3d0ffa3121fe974b73f068c8282cb1c32328113b6c"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"futures-util",
|
"futures-util",
|
||||||
"log 0.4.21",
|
"log 0.4.21",
|
||||||
"rustls 0.21.10",
|
"rustls 0.21.12",
|
||||||
"tokio",
|
"tokio",
|
||||||
"tokio-rustls 0.24.1",
|
"tokio-rustls 0.24.1",
|
||||||
"tungstenite 0.20.1",
|
"tungstenite 0.20.1",
|
||||||
|
@ -9264,7 +9300,7 @@ dependencies = [
|
||||||
"percent-encoding 2.3.1",
|
"percent-encoding 2.3.1",
|
||||||
"pin-project",
|
"pin-project",
|
||||||
"prost 0.12.4",
|
"prost 0.12.4",
|
||||||
"rustls 0.21.10",
|
"rustls 0.21.12",
|
||||||
"rustls-native-certs",
|
"rustls-native-certs",
|
||||||
"rustls-pemfile",
|
"rustls-pemfile",
|
||||||
"tokio",
|
"tokio",
|
||||||
|
@ -9480,7 +9516,7 @@ dependencies = [
|
||||||
"httparse",
|
"httparse",
|
||||||
"log 0.4.21",
|
"log 0.4.21",
|
||||||
"rand 0.8.5",
|
"rand 0.8.5",
|
||||||
"rustls 0.21.10",
|
"rustls 0.21.12",
|
||||||
"sha1",
|
"sha1",
|
||||||
"thiserror",
|
"thiserror",
|
||||||
"url 2.5.0",
|
"url 2.5.0",
|
||||||
|
|
|
@ -17,6 +17,7 @@ solana-rpc-client-api = { version = "1.17" }
|
||||||
mango-feeds-connector = { git = "https://github.com/blockworks-foundation/mango-feeds.git", tag = "connector-v0.4.8" }
|
mango-feeds-connector = { git = "https://github.com/blockworks-foundation/mango-feeds.git", tag = "connector-v0.4.8" }
|
||||||
yellowstone-grpc-client = { version = "1.15.0", git = "https://github.com/blockworks-foundation/yellowstone-grpc.git", tag = "v1.15.0+solana.1.17" }
|
yellowstone-grpc-client = { version = "1.15.0", git = "https://github.com/blockworks-foundation/yellowstone-grpc.git", tag = "v1.15.0+solana.1.17" }
|
||||||
yellowstone-grpc-proto = { version = "1.14.0", git = "https://github.com/blockworks-foundation/yellowstone-grpc.git", tag = "v1.15.0+solana.1.17" }
|
yellowstone-grpc-proto = { version = "1.14.0", git = "https://github.com/blockworks-foundation/yellowstone-grpc.git", tag = "v1.15.0+solana.1.17" }
|
||||||
|
|
||||||
reqwest = { version = "0.11.27", features = ["json"] }
|
reqwest = { version = "0.11.27", features = ["json"] }
|
||||||
whirlpools-client = { git = "https://github.com/blockworks-foundation/whirlpools-client/", features = ["no-entrypoint"] }
|
whirlpools-client = { git = "https://github.com/blockworks-foundation/whirlpools-client/", features = ["no-entrypoint"] }
|
||||||
openbook-v2 = { git = "https://github.com/openbook-dex/openbook-v2", tag = "v0.2.7", features = ["no-entrypoint", "client"] }
|
openbook-v2 = { git = "https://github.com/openbook-dex/openbook-v2", tag = "v0.2.7", features = ["no-entrypoint", "client"] }
|
||||||
|
@ -25,6 +26,8 @@ stable-swap = { version = "1.8.1", features = ["no-entrypoint", "client"] }
|
||||||
stable-swap-client = { version = "1.8.1" }
|
stable-swap-client = { version = "1.8.1" }
|
||||||
stable-swap-math = { version = "1.8.1" }
|
stable-swap-math = { version = "1.8.1" }
|
||||||
uint = { version = "0.9.1" }
|
uint = { version = "0.9.1" }
|
||||||
|
quic-geyser-client = { git = "https://github.com/blockworks-foundation/quic_geyser_plugin.git", branch = "router_v1.17.29" }
|
||||||
|
quic-geyser-common = { git = "https://github.com/blockworks-foundation/quic_geyser_plugin.git", branch = "router_v1.17.29" }
|
||||||
|
|
||||||
[profile.release]
|
[profile.release]
|
||||||
overflow-checks = true
|
overflow-checks = true
|
||||||
|
|
|
@ -87,6 +87,10 @@ yellowstone-grpc-client = { workspace = true }
|
||||||
yellowstone-grpc-proto = { workspace = true }
|
yellowstone-grpc-proto = { workspace = true }
|
||||||
tonic = { version = "0.10.2", features = ["gzip"] }
|
tonic = { version = "0.10.2", features = ["gzip"] }
|
||||||
|
|
||||||
|
# quic
|
||||||
|
quic-geyser-client = { workspace = true }
|
||||||
|
quic-geyser-common = { workspace = true }
|
||||||
|
|
||||||
# compressed snapshots
|
# compressed snapshots
|
||||||
lz4 = "1.24.0"
|
lz4 = "1.24.0"
|
||||||
|
|
||||||
|
|
|
@ -22,13 +22,13 @@ pub async fn main() {
|
||||||
let rpc_http_addr = env::var("RPC_HTTP_ADDR").expect("need rpc http url");
|
let rpc_http_addr = env::var("RPC_HTTP_ADDR").expect("need rpc http url");
|
||||||
let snapshot_config = AccountDataSourceConfig {
|
let snapshot_config = AccountDataSourceConfig {
|
||||||
region: None,
|
region: None,
|
||||||
use_quic: None,
|
quic_sources: None,
|
||||||
quic_address: None,
|
|
||||||
rpc_http_url: rpc_http_addr.clone(),
|
rpc_http_url: rpc_http_addr.clone(),
|
||||||
rpc_support_compression: Some(false), /* no compression */
|
rpc_support_compression: Some(false), /* no compression */
|
||||||
re_snapshot_interval_secs: None,
|
re_snapshot_interval_secs: None,
|
||||||
grpc_sources: vec![],
|
grpc_sources: Some(vec![]),
|
||||||
dedup_queue_size: 0,
|
dedup_queue_size: 0,
|
||||||
|
request_timeout_in_seconds: None,
|
||||||
};
|
};
|
||||||
|
|
||||||
// Raydium
|
// Raydium
|
||||||
|
|
|
@ -138,7 +138,8 @@ pub fn spawn_updater_job(
|
||||||
),
|
),
|
||||||
};
|
};
|
||||||
|
|
||||||
let snapshot_timeout = Instant::now() + Duration::from_secs(60 * 5);
|
let snapshot_timeout_in_seconds = config.snapshot_timeout_in_seconds.unwrap_or(60 * 5);
|
||||||
|
let snapshot_timeout = Instant::now() + Duration::from_secs(snapshot_timeout_in_seconds);
|
||||||
let listener_job = tokio_spawn(format!("edge_updater_{}", dex.name).as_str(), async move {
|
let listener_job = tokio_spawn(format!("edge_updater_{}", dex.name).as_str(), async move {
|
||||||
let mut updater = EdgeUpdater {
|
let mut updater = EdgeUpdater {
|
||||||
dex,
|
dex,
|
||||||
|
|
|
@ -175,19 +175,28 @@ async fn main() -> anyhow::Result<()> {
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
if source_config.grpc_sources.len() > 1 {
|
if let Some(quic_sources) = &source_config.quic_sources {
|
||||||
error!("only one grpc source is supported ATM");
|
info!(
|
||||||
exit(-1);
|
"quic sources: {}",
|
||||||
|
quic_sources
|
||||||
|
.iter()
|
||||||
|
.map(|c| c.connection_string.clone())
|
||||||
|
.collect::<String>()
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
if let Some(grpc_sources) = source_config.grpc_sources.clone() {
|
||||||
info!(
|
info!(
|
||||||
"grpc sources: {}",
|
"grpc sources: {}",
|
||||||
source_config
|
grpc_sources
|
||||||
.grpc_sources
|
.iter()
|
||||||
.iter()
|
.map(|c| c.connection_string.clone())
|
||||||
.map(|c| c.connection_string.clone())
|
.collect::<String>()
|
||||||
.collect::<String>()
|
);
|
||||||
);
|
} else {
|
||||||
|
// current grpc source is needed for transaction watcher even if there is quic
|
||||||
|
error!("No grpc geyser sources specified");
|
||||||
|
exit(-1);
|
||||||
|
};
|
||||||
|
|
||||||
if config.metrics.output_http {
|
if config.metrics.output_http {
|
||||||
let prom_bind_addr = config
|
let prom_bind_addr = config
|
||||||
|
@ -443,8 +452,8 @@ async fn main() -> anyhow::Result<()> {
|
||||||
let ef = exit_sender.subscribe();
|
let ef = exit_sender.subscribe();
|
||||||
let sc = source_config.clone();
|
let sc = source_config.clone();
|
||||||
let account_update_job = tokio_spawn("geyser", async move {
|
let account_update_job = tokio_spawn("geyser", async move {
|
||||||
if sc.use_quic.unwrap_or(false) {
|
if sc.grpc_sources.is_none() && sc.quic_sources.is_none() {
|
||||||
error!("not supported yet");
|
error!("No quic or grpc plugin setup");
|
||||||
} else {
|
} else {
|
||||||
geyser::spawn_geyser_source(
|
geyser::spawn_geyser_source(
|
||||||
&sc,
|
&sc,
|
||||||
|
@ -528,7 +537,7 @@ fn build_price_feed(
|
||||||
fn build_rpc(source_config: &AccountDataSourceConfig) -> RpcClient {
|
fn build_rpc(source_config: &AccountDataSourceConfig) -> RpcClient {
|
||||||
RpcClient::new_with_timeouts_and_commitment(
|
RpcClient::new_with_timeouts_and_commitment(
|
||||||
string_or_env(source_config.rpc_http_url.clone()),
|
string_or_env(source_config.rpc_http_url.clone()),
|
||||||
Duration::from_secs(60), // request timeout
|
Duration::from_secs(source_config.request_timeout_in_seconds.unwrap_or(60)), // request timeout
|
||||||
CommitmentConfig::confirmed(),
|
CommitmentConfig::confirmed(),
|
||||||
Duration::from_secs(60), // confirmation timeout
|
Duration::from_secs(60), // confirmation timeout
|
||||||
)
|
)
|
||||||
|
@ -537,7 +546,7 @@ fn build_rpc(source_config: &AccountDataSourceConfig) -> RpcClient {
|
||||||
fn build_blocking_rpc(source_config: &AccountDataSourceConfig) -> BlockingRpcClient {
|
fn build_blocking_rpc(source_config: &AccountDataSourceConfig) -> BlockingRpcClient {
|
||||||
BlockingRpcClient::new_with_timeouts_and_commitment(
|
BlockingRpcClient::new_with_timeouts_and_commitment(
|
||||||
string_or_env(source_config.rpc_http_url.clone()),
|
string_or_env(source_config.rpc_http_url.clone()),
|
||||||
Duration::from_secs(60), // request timeout
|
Duration::from_secs(source_config.request_timeout_in_seconds.unwrap_or(60)), // request timeout
|
||||||
CommitmentConfig::confirmed(),
|
CommitmentConfig::confirmed(),
|
||||||
Duration::from_secs(60), // confirmation timeout
|
Duration::from_secs(60), // confirmation timeout
|
||||||
)
|
)
|
||||||
|
|
|
@ -27,6 +27,23 @@ lazy_static::lazy_static! {
|
||||||
pub static ref GRPC_TO_EDGE_SLOT_LAG: IntGaugeVec =
|
pub static ref GRPC_TO_EDGE_SLOT_LAG: IntGaugeVec =
|
||||||
register_int_gauge_vec!(opts!("router_grpc_to_edge_slot_lag", "RPC Slot vs last slot used to update edges"), &["dex_name"]).unwrap();
|
register_int_gauge_vec!(opts!("router_grpc_to_edge_slot_lag", "RPC Slot vs last slot used to update edges"), &["dex_name"]).unwrap();
|
||||||
|
|
||||||
|
pub static ref QUIC_ACCOUNT_WRITES: IntCounter =
|
||||||
|
register_int_counter!("quic_account_writes", "Number of account updates via Geyser gRPC").unwrap();
|
||||||
|
pub static ref QUIC_ACCOUNT_WRITE_QUEUE: IntGauge =
|
||||||
|
register_int_gauge!("quic_account_write_queue", "Items in account write queue via Geyser gPRC").unwrap();
|
||||||
|
pub static ref QUIC_DEDUP_QUEUE: GenericGauge<prometheus::core::AtomicI64> =
|
||||||
|
register_int_gauge!("quic_dedup_queue", "Items in dedup queue via Geyser gPRC").unwrap();
|
||||||
|
pub static ref QUIC_SLOT_UPDATE_QUEUE: GenericGauge<prometheus::core::AtomicI64> =
|
||||||
|
register_int_gauge!("quic_slot_update_queue", "Items in slot update queue via Geyser gPRC").unwrap();
|
||||||
|
pub static ref QUIC_SLOT_UPDATES: IntCounter =
|
||||||
|
register_int_counter!("quic_slot_updates", "Number of slot updates via Geyser gPRC").unwrap();
|
||||||
|
pub static ref QUIC_SNAPSHOT_ACCOUNT_WRITES: IntCounter =
|
||||||
|
register_int_counter!("quic_snapshot_account_writes", "Number of account writes from snapshot").unwrap();
|
||||||
|
pub static ref QUIC_SOURCE_CONNECTION_RETRIES: IntCounterVec =
|
||||||
|
register_int_counter_vec!(opts!("quic_source_connection_retries", "gRPC source connection retries"), &["source_name"]).unwrap();
|
||||||
|
pub static ref QUIC_NO_MESSAGE_FOR_DURATION_MS: IntGauge =
|
||||||
|
register_int_gauge!("quic_no_update_for_duration_ms", "Did not get any message from Geyser gPRC for this duration").unwrap();
|
||||||
|
|
||||||
pub static ref HTTP_REQUEST_TIMING: HistogramVec =
|
pub static ref HTTP_REQUEST_TIMING: HistogramVec =
|
||||||
register_histogram_vec!(
|
register_histogram_vec!(
|
||||||
histogram_opts!("router_http_request_timing", "Endpoint timing in seconds",
|
histogram_opts!("router_http_request_timing", "Endpoint timing in seconds",
|
||||||
|
|
|
@ -9,6 +9,8 @@ use router_feed_lib::get_program_account::FeedMetadata;
|
||||||
|
|
||||||
use crate::source::grpc_plugin_source;
|
use crate::source::grpc_plugin_source;
|
||||||
|
|
||||||
|
use super::quic_plugin_source;
|
||||||
|
|
||||||
pub async fn spawn_geyser_source(
|
pub async fn spawn_geyser_source(
|
||||||
config: &AccountDataSourceConfig,
|
config: &AccountDataSourceConfig,
|
||||||
exit_receiver: tokio::sync::broadcast::Receiver<()>,
|
exit_receiver: tokio::sync::broadcast::Receiver<()>,
|
||||||
|
@ -20,16 +22,31 @@ pub async fn spawn_geyser_source(
|
||||||
subscribed_token_accounts: &HashSet<Pubkey>,
|
subscribed_token_accounts: &HashSet<Pubkey>,
|
||||||
filters: &HashSet<Pubkey>,
|
filters: &HashSet<Pubkey>,
|
||||||
) {
|
) {
|
||||||
grpc_plugin_source::process_events(
|
if config.quic_sources.is_some() {
|
||||||
config.clone(),
|
quic_plugin_source::process_events(
|
||||||
subscribed_accounts.clone(),
|
config.clone(),
|
||||||
subscribed_programs.clone(),
|
subscribed_accounts.clone(),
|
||||||
subscribed_token_accounts.clone(),
|
subscribed_programs.clone(),
|
||||||
filters.clone(),
|
subscribed_token_accounts.clone(),
|
||||||
account_write_sender,
|
filters.clone(),
|
||||||
Some(metadata_write_sender),
|
account_write_sender,
|
||||||
slot_sender,
|
Some(metadata_write_sender),
|
||||||
exit_receiver,
|
slot_sender,
|
||||||
)
|
exit_receiver,
|
||||||
.await;
|
)
|
||||||
|
.await;
|
||||||
|
} else if config.grpc_sources.is_some() {
|
||||||
|
grpc_plugin_source::process_events(
|
||||||
|
config.clone(),
|
||||||
|
subscribed_accounts.clone(),
|
||||||
|
subscribed_programs.clone(),
|
||||||
|
subscribed_token_accounts.clone(),
|
||||||
|
filters.clone(),
|
||||||
|
account_write_sender,
|
||||||
|
Some(metadata_write_sender),
|
||||||
|
slot_sender,
|
||||||
|
exit_receiver,
|
||||||
|
)
|
||||||
|
.await;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -515,13 +515,13 @@ pub async fn process_events(
|
||||||
async_channel::bounded::<SourceMessage>(config.dedup_queue_size);
|
async_channel::bounded::<SourceMessage>(config.dedup_queue_size);
|
||||||
let mut source_jobs = vec![];
|
let mut source_jobs = vec![];
|
||||||
|
|
||||||
|
let Some(grpc_sources) = config.grpc_sources.clone() else {
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
|
||||||
// note: caller in main.rs ensures this
|
// note: caller in main.rs ensures this
|
||||||
assert_eq!(
|
assert_eq!(grpc_sources.len(), 1, "only one grpc source supported");
|
||||||
config.grpc_sources.len(),
|
for grpc_source in grpc_sources.clone() {
|
||||||
1,
|
|
||||||
"only one grpc source supported"
|
|
||||||
);
|
|
||||||
for grpc_source in config.grpc_sources.clone() {
|
|
||||||
let msg_sender = msg_sender.clone();
|
let msg_sender = msg_sender.clone();
|
||||||
let sub_accounts = subscription_accounts.clone();
|
let sub_accounts = subscription_accounts.clone();
|
||||||
let sub_programs = subscription_programs.clone();
|
let sub_programs = subscription_programs.clone();
|
||||||
|
|
|
@ -88,7 +88,6 @@ pub async fn request_mint_metadata(
|
||||||
count.fetch_add(1, Ordering::Relaxed);
|
count.fetch_add(1, Ordering::Relaxed);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
mint_accounts
|
mint_accounts
|
||||||
});
|
});
|
||||||
threads.push(jh_thread);
|
threads.push(jh_thread);
|
||||||
|
|
|
@ -1,3 +1,4 @@
|
||||||
pub mod geyser;
|
pub mod geyser;
|
||||||
pub mod grpc_plugin_source;
|
pub mod grpc_plugin_source;
|
||||||
pub mod mint_accounts_source;
|
pub mod mint_accounts_source;
|
||||||
|
pub mod quic_plugin_source;
|
||||||
|
|
|
@ -0,0 +1,605 @@
|
||||||
|
use itertools::Itertools;
|
||||||
|
use jsonrpc_core::futures::StreamExt;
|
||||||
|
|
||||||
|
use quic_geyser_common::filters::MemcmpFilter;
|
||||||
|
use quic_geyser_common::types::connections_parameters::ConnectionParameters;
|
||||||
|
use solana_sdk::pubkey::Pubkey;
|
||||||
|
|
||||||
|
use anchor_spl::token::spl_token;
|
||||||
|
use async_channel::{Receiver, Sender};
|
||||||
|
use std::collections::HashSet;
|
||||||
|
use std::str::FromStr;
|
||||||
|
use std::sync::Arc;
|
||||||
|
use std::time::Instant;
|
||||||
|
use std::{collections::HashMap, env, time::Duration};
|
||||||
|
use tracing::*;
|
||||||
|
|
||||||
|
use crate::metrics;
|
||||||
|
use mango_feeds_connector::{chain_data::SlotStatus, SlotUpdate};
|
||||||
|
use quic_geyser_common::message::Message;
|
||||||
|
use router_config_lib::{AccountDataSourceConfig, QuicSourceConfig};
|
||||||
|
use router_feed_lib::account_write::{AccountOrSnapshotUpdate, AccountWrite};
|
||||||
|
use router_feed_lib::get_program_account::{
|
||||||
|
get_snapshot_gma, get_snapshot_gpa, get_snapshot_gta, CustomSnapshotProgramAccounts,
|
||||||
|
FeedMetadata,
|
||||||
|
};
|
||||||
|
use solana_program::clock::Slot;
|
||||||
|
use tokio::sync::Semaphore;
|
||||||
|
|
||||||
|
const MAX_GMA_ACCOUNTS: usize = 100;
|
||||||
|
|
||||||
|
// limit number of concurrent gMA/gPA requests
|
||||||
|
const MAX_PARALLEL_HEAVY_RPC_REQUESTS: usize = 4;
|
||||||
|
|
||||||
|
#[allow(clippy::large_enum_variant)]
|
||||||
|
pub enum SourceMessage {
|
||||||
|
QuicMessage(Message),
|
||||||
|
Snapshot(CustomSnapshotProgramAccounts),
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn feed_data_geyser(
|
||||||
|
quic_source_config: &QuicSourceConfig,
|
||||||
|
snapshot_config: AccountDataSourceConfig,
|
||||||
|
subscribed_accounts: &HashSet<Pubkey>,
|
||||||
|
subscribed_programs: &HashSet<Pubkey>,
|
||||||
|
subscribed_token_accounts: &HashSet<Pubkey>,
|
||||||
|
sender: async_channel::Sender<SourceMessage>,
|
||||||
|
) -> anyhow::Result<()> {
|
||||||
|
let use_compression = snapshot_config.rpc_support_compression.unwrap_or(false);
|
||||||
|
|
||||||
|
let snapshot_rpc_http_url = match &snapshot_config.rpc_http_url.chars().next().unwrap() {
|
||||||
|
'$' => env::var(&snapshot_config.rpc_http_url[1..])
|
||||||
|
.expect("reading connection string from env"),
|
||||||
|
_ => snapshot_config.rpc_http_url.clone(),
|
||||||
|
};
|
||||||
|
info!("connecting to quic source {:?}", quic_source_config);
|
||||||
|
|
||||||
|
let (quic_client, mut stream, _jh) = quic_geyser_client::non_blocking::client::Client::new(
|
||||||
|
quic_source_config.connection_string.clone(),
|
||||||
|
ConnectionParameters {
|
||||||
|
enable_gso: quic_source_config.enable_gso.unwrap_or(true),
|
||||||
|
..Default::default()
|
||||||
|
},
|
||||||
|
)
|
||||||
|
.await?;
|
||||||
|
|
||||||
|
let mut subscriptions = vec![];
|
||||||
|
|
||||||
|
let subscribed_program_filter = subscribed_programs.iter().map(|x| {
|
||||||
|
quic_geyser_common::filters::Filter::Account(quic_geyser_common::filters::AccountFilter {
|
||||||
|
owner: Some(*x),
|
||||||
|
accounts: None,
|
||||||
|
filters: None,
|
||||||
|
})
|
||||||
|
});
|
||||||
|
subscriptions.extend(subscribed_program_filter);
|
||||||
|
|
||||||
|
let subscribed_token_accounts_filter = subscribed_programs.iter().map(|x| {
|
||||||
|
quic_geyser_common::filters::Filter::Account(quic_geyser_common::filters::AccountFilter {
|
||||||
|
owner: Some(Pubkey::from_str("TokenkegQfeZyiNwAJbNbGKPFXCWuBvf9Ss623VQ5DA").unwrap()),
|
||||||
|
accounts: None,
|
||||||
|
filters: Some(vec![
|
||||||
|
quic_geyser_common::filters::AccountFilterType::Datasize(165),
|
||||||
|
quic_geyser_common::filters::AccountFilterType::Memcmp(MemcmpFilter {
|
||||||
|
offset: 32,
|
||||||
|
data: quic_geyser_common::filters::MemcmpFilterData::Bytes(
|
||||||
|
x.to_bytes().to_vec(),
|
||||||
|
),
|
||||||
|
}),
|
||||||
|
]),
|
||||||
|
})
|
||||||
|
});
|
||||||
|
subscriptions.extend(subscribed_token_accounts_filter);
|
||||||
|
|
||||||
|
subscriptions.push(quic_geyser_common::filters::Filter::Account(
|
||||||
|
quic_geyser_common::filters::AccountFilter {
|
||||||
|
accounts: Some(subscribed_accounts.clone()),
|
||||||
|
owner: None,
|
||||||
|
filters: None,
|
||||||
|
},
|
||||||
|
));
|
||||||
|
|
||||||
|
subscriptions.push(quic_geyser_common::filters::Filter::Slot);
|
||||||
|
quic_client.subscribe(subscriptions).await?;
|
||||||
|
|
||||||
|
// We can't get a snapshot immediately since the finalized snapshot would be for a
|
||||||
|
// slot in the past and we'd be missing intermediate updates.
|
||||||
|
//
|
||||||
|
// Delay the request until the first slot we received all writes for becomes rooted
|
||||||
|
// to avoid that problem - partially. The rooted slot will still be larger than the
|
||||||
|
// finalized slot, so add a number of slots as a buffer.
|
||||||
|
//
|
||||||
|
// If that buffer isn't sufficient, there'll be a retry.
|
||||||
|
|
||||||
|
// The first slot that we will receive _all_ account writes for
|
||||||
|
let mut first_full_slot: u64 = u64::MAX;
|
||||||
|
|
||||||
|
// If a snapshot should be performed when ready.
|
||||||
|
let mut snapshot_needed = true;
|
||||||
|
|
||||||
|
// The highest "rooted" slot that has been seen.
|
||||||
|
let mut max_finalized_slot = 0;
|
||||||
|
|
||||||
|
// Data for slots will arrive out of order. This value defines how many
|
||||||
|
// slots after a slot was marked "rooted" we assume it'll not receive
|
||||||
|
// any more account write information.
|
||||||
|
//
|
||||||
|
// This is important for the write_version mapping (to know when slots can
|
||||||
|
// be dropped).
|
||||||
|
let max_out_of_order_slots = 40;
|
||||||
|
|
||||||
|
// Number of slots that we expect "finalized" commitment to lag
|
||||||
|
// behind "rooted". This matters for getProgramAccounts based snapshots,
|
||||||
|
// which will have "finalized" commitment.
|
||||||
|
let mut rooted_to_finalized_slots = 30;
|
||||||
|
|
||||||
|
let (snapshot_gma_sender, mut snapshot_gma_receiver) = tokio::sync::mpsc::unbounded_channel();
|
||||||
|
// TODO log buffer size
|
||||||
|
|
||||||
|
// The plugin sends a ping every 5s or so
|
||||||
|
let fatal_idle_timeout = Duration::from_secs(15);
|
||||||
|
let mut re_snapshot_interval = tokio::time::interval(Duration::from_secs(
|
||||||
|
snapshot_config
|
||||||
|
.re_snapshot_interval_secs
|
||||||
|
.unwrap_or(60 * 60 * 12),
|
||||||
|
));
|
||||||
|
re_snapshot_interval.set_missed_tick_behavior(tokio::time::MissedTickBehavior::Delay);
|
||||||
|
re_snapshot_interval.tick().await;
|
||||||
|
|
||||||
|
// Highest slot that an account write came in for.
|
||||||
|
let mut newest_write_slot: u64 = 0;
|
||||||
|
|
||||||
|
let mut last_message_received_at = Instant::now();
|
||||||
|
|
||||||
|
loop {
|
||||||
|
tokio::select! {
|
||||||
|
update = stream.recv() => {
|
||||||
|
let Some(mut message) = update
|
||||||
|
else {
|
||||||
|
anyhow::bail!("geyser plugin has closed the stream");
|
||||||
|
};
|
||||||
|
// use account and slot updates to trigger snapshot loading
|
||||||
|
match &mut message {
|
||||||
|
Message::SlotMsg(slot_update) => {
|
||||||
|
trace!("received slot update for slot {}", slot_update.slot);
|
||||||
|
let commitment_config = slot_update.commitment_config;
|
||||||
|
|
||||||
|
debug!(
|
||||||
|
"slot_update: {} ({:?})",
|
||||||
|
slot_update.slot,
|
||||||
|
commitment_config
|
||||||
|
);
|
||||||
|
|
||||||
|
if commitment_config.is_finalized() {
|
||||||
|
if first_full_slot == u64::MAX {
|
||||||
|
// TODO: is this equivalent to before? what was highesy_write_slot?
|
||||||
|
first_full_slot = slot_update.slot + 1;
|
||||||
|
}
|
||||||
|
// TODO rename rooted to finalized
|
||||||
|
if slot_update.slot > max_finalized_slot {
|
||||||
|
max_finalized_slot = slot_update.slot;
|
||||||
|
}
|
||||||
|
|
||||||
|
let waiting_for_snapshot_slot = max_finalized_slot <= first_full_slot + rooted_to_finalized_slots;
|
||||||
|
|
||||||
|
if waiting_for_snapshot_slot {
|
||||||
|
debug!("waiting for snapshot slot: rooted={}, first_full={}, slot={}", max_finalized_slot, first_full_slot, slot_update.slot);
|
||||||
|
}
|
||||||
|
|
||||||
|
if snapshot_needed && !waiting_for_snapshot_slot {
|
||||||
|
snapshot_needed = false;
|
||||||
|
|
||||||
|
debug!("snapshot slot reached - setting up snapshot tasks");
|
||||||
|
|
||||||
|
let permits_parallel_rpc_requests = Arc::new(Semaphore::new(MAX_PARALLEL_HEAVY_RPC_REQUESTS));
|
||||||
|
|
||||||
|
info!("Requesting snapshot from gMA for {} filter accounts", subscribed_accounts.len());
|
||||||
|
for pubkey_chunk in subscribed_accounts.iter().chunks(MAX_GMA_ACCOUNTS).into_iter() {
|
||||||
|
let rpc_http_url = snapshot_rpc_http_url.clone();
|
||||||
|
let account_ids = pubkey_chunk.map(|x| *x).collect_vec();
|
||||||
|
let sender = snapshot_gma_sender.clone();
|
||||||
|
let permits = permits_parallel_rpc_requests.clone();
|
||||||
|
tokio::spawn(async move {
|
||||||
|
let _permit = permits.acquire().await.unwrap();
|
||||||
|
let snapshot = get_snapshot_gma(&rpc_http_url, &account_ids).await;
|
||||||
|
match sender.send(snapshot) {
|
||||||
|
Ok(_) => {}
|
||||||
|
Err(_) => {
|
||||||
|
warn!("Could not send snapshot, quic has probably reconnected");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
info!("Requesting snapshot from gPA for {} program filter accounts", subscribed_programs.len());
|
||||||
|
for program_id in subscribed_programs {
|
||||||
|
let rpc_http_url = snapshot_rpc_http_url.clone();
|
||||||
|
let program_id = *program_id;
|
||||||
|
let sender = snapshot_gma_sender.clone();
|
||||||
|
let permits = permits_parallel_rpc_requests.clone();
|
||||||
|
tokio::spawn(async move {
|
||||||
|
let _permit = permits.acquire().await.unwrap();
|
||||||
|
let snapshot = get_snapshot_gpa(&rpc_http_url, &program_id, use_compression).await;
|
||||||
|
match sender.send(snapshot) {
|
||||||
|
Ok(_) => {}
|
||||||
|
Err(_) => {
|
||||||
|
warn!("Could not send snapshot, quic has probably reconnected");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
info!("Requesting snapshot from gTA for {} owners filter accounts", subscribed_token_accounts.len());
|
||||||
|
for owner_id in subscribed_token_accounts {
|
||||||
|
let rpc_http_url = snapshot_rpc_http_url.clone();
|
||||||
|
let owner_id = owner_id.clone();
|
||||||
|
let sender = snapshot_gma_sender.clone();
|
||||||
|
let permits = permits_parallel_rpc_requests.clone();
|
||||||
|
tokio::spawn(async move {
|
||||||
|
let _permit = permits.acquire().await.unwrap();
|
||||||
|
let snapshot = get_snapshot_gta(&rpc_http_url, &owner_id).await;
|
||||||
|
match sender.send(snapshot) {
|
||||||
|
Ok(_) => {}
|
||||||
|
Err(_) => {
|
||||||
|
warn!("Could not send snapshot, quic has probably reconnected");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
Message::AccountMsg(info) => {
|
||||||
|
let slot = info.slot_identifier.slot;
|
||||||
|
trace!("received account update for slot {}", slot);
|
||||||
|
if slot < first_full_slot {
|
||||||
|
// Don't try to process data for slots where we may have missed writes:
|
||||||
|
// We could not map the write_version correctly for them.
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if slot > newest_write_slot {
|
||||||
|
newest_write_slot = slot;
|
||||||
|
debug!(
|
||||||
|
"newest_write_slot: {}",
|
||||||
|
newest_write_slot
|
||||||
|
);
|
||||||
|
} else if max_finalized_slot > 0 && info.slot_identifier.slot < max_finalized_slot - max_out_of_order_slots {
|
||||||
|
anyhow::bail!("received write {} slots back from max rooted slot {}", max_finalized_slot - slot, max_finalized_slot);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
_ => {
|
||||||
|
// ignore all other quic update types
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let elapsed = last_message_received_at.elapsed().as_millis();
|
||||||
|
metrics::QUIC_NO_MESSAGE_FOR_DURATION_MS.set(elapsed as i64);
|
||||||
|
last_message_received_at = Instant::now();
|
||||||
|
|
||||||
|
// send the incremental updates to the channel
|
||||||
|
sender.send(SourceMessage::QuicMessage(message)).await.expect("send success");
|
||||||
|
},
|
||||||
|
snapshot_message = snapshot_gma_receiver.recv() => {
|
||||||
|
let Some(snapshot_result) = snapshot_message
|
||||||
|
else {
|
||||||
|
anyhow::bail!("snapshot channel closed");
|
||||||
|
};
|
||||||
|
let snapshot = snapshot_result?;
|
||||||
|
debug!("snapshot (program={}, m_accounts={}) is for slot {}, first full slot was {}",
|
||||||
|
snapshot.program_id.map(|x| x.to_string()).unwrap_or("none".to_string()),
|
||||||
|
snapshot.accounts.len(),
|
||||||
|
snapshot.slot,
|
||||||
|
first_full_slot);
|
||||||
|
|
||||||
|
if snapshot.slot < first_full_slot {
|
||||||
|
warn!(
|
||||||
|
"snapshot is too old: has slot {}, expected {} minimum - request another one but also use this snapshot",
|
||||||
|
snapshot.slot,
|
||||||
|
first_full_slot
|
||||||
|
);
|
||||||
|
// try again in another 25 slots
|
||||||
|
snapshot_needed = true;
|
||||||
|
rooted_to_finalized_slots += 25;
|
||||||
|
}
|
||||||
|
|
||||||
|
// New - Don't care if the snapshot is old, we want startup to work anyway
|
||||||
|
// If an edge is not working properly, it will be disabled when swapping it
|
||||||
|
sender
|
||||||
|
.send(SourceMessage::Snapshot(snapshot))
|
||||||
|
.await
|
||||||
|
.expect("send success");
|
||||||
|
|
||||||
|
},
|
||||||
|
_ = tokio::time::sleep(fatal_idle_timeout) => {
|
||||||
|
anyhow::bail!("geyser plugin hasn't sent a message in too long");
|
||||||
|
}
|
||||||
|
_ = re_snapshot_interval.tick() => {
|
||||||
|
info!("Re-snapshot hack");
|
||||||
|
snapshot_needed = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn process_events(
|
||||||
|
config: AccountDataSourceConfig,
|
||||||
|
subscription_accounts: HashSet<Pubkey>,
|
||||||
|
subscription_programs: HashSet<Pubkey>,
|
||||||
|
subscription_token_accounts: HashSet<Pubkey>,
|
||||||
|
filters: HashSet<Pubkey>,
|
||||||
|
account_write_queue_sender: async_channel::Sender<AccountOrSnapshotUpdate>,
|
||||||
|
metdata_write_queue_sender: Option<async_channel::Sender<FeedMetadata>>,
|
||||||
|
slot_queue_sender: async_channel::Sender<SlotUpdate>,
|
||||||
|
mut exit: tokio::sync::broadcast::Receiver<()>,
|
||||||
|
) {
|
||||||
|
// Subscribe to geyser
|
||||||
|
let (msg_sender, msg_receiver) =
|
||||||
|
async_channel::bounded::<SourceMessage>(config.dedup_queue_size);
|
||||||
|
let mut source_jobs = vec![];
|
||||||
|
|
||||||
|
let Some(quic_sources) = config.quic_sources.clone() else {
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
|
||||||
|
// note: caller in main.rs ensures this
|
||||||
|
assert_eq!(quic_sources.len(), 1, "only one quic source supported");
|
||||||
|
for quic_source in quic_sources.clone() {
|
||||||
|
let msg_sender = msg_sender.clone();
|
||||||
|
let sub_accounts = subscription_accounts.clone();
|
||||||
|
let sub_programs = subscription_programs.clone();
|
||||||
|
let sub_token_accounts = subscription_token_accounts.clone();
|
||||||
|
|
||||||
|
let cfg = config.clone();
|
||||||
|
|
||||||
|
source_jobs.push(tokio::spawn(async move {
|
||||||
|
let mut error_count = 0;
|
||||||
|
let mut last_error = Instant::now();
|
||||||
|
|
||||||
|
// Continuously reconnect on failure
|
||||||
|
loop {
|
||||||
|
let out = feed_data_geyser(
|
||||||
|
&quic_source,
|
||||||
|
cfg.clone(),
|
||||||
|
&sub_accounts,
|
||||||
|
&sub_programs,
|
||||||
|
&sub_token_accounts,
|
||||||
|
msg_sender.clone(),
|
||||||
|
);
|
||||||
|
if last_error.elapsed() > Duration::from_secs(60 * 10) {
|
||||||
|
error_count = 0;
|
||||||
|
}
|
||||||
|
else if error_count > 10 {
|
||||||
|
error!("error during communication with the geyser plugin - retried too many time, exiting..");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
match out.await {
|
||||||
|
// happy case!
|
||||||
|
Err(err) => {
|
||||||
|
warn!(
|
||||||
|
"error during communication with the geyser plugin - retrying: {:?}",
|
||||||
|
err
|
||||||
|
);
|
||||||
|
last_error = Instant::now();
|
||||||
|
error_count += 1;
|
||||||
|
}
|
||||||
|
// this should never happen
|
||||||
|
Ok(_) => {
|
||||||
|
error!("feed_data must return an error, not OK - continue");
|
||||||
|
last_error = Instant::now();
|
||||||
|
error_count += 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
metrics::QUIC_SOURCE_CONNECTION_RETRIES
|
||||||
|
.with_label_values(&[&quic_source.name])
|
||||||
|
.inc();
|
||||||
|
|
||||||
|
tokio::time::sleep(std::time::Duration::from_secs(
|
||||||
|
quic_source.retry_connection_sleep_secs,
|
||||||
|
))
|
||||||
|
.await;
|
||||||
|
}
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
|
// slot -> (pubkey -> write_version)
|
||||||
|
//
|
||||||
|
// To avoid unnecessarily sending requests to SQL, we track the latest write_version
|
||||||
|
// for each (slot, pubkey). If an already-seen write_version comes in, it can be safely
|
||||||
|
// discarded.
|
||||||
|
let mut latest_write = HashMap::<Slot, HashMap<Pubkey, u64>>::new();
|
||||||
|
|
||||||
|
// Number of slots to retain in latest_write
|
||||||
|
let latest_write_retention = 50;
|
||||||
|
|
||||||
|
let mut source_jobs: futures::stream::FuturesUnordered<_> = source_jobs.into_iter().collect();
|
||||||
|
|
||||||
|
loop {
|
||||||
|
tokio::select! {
|
||||||
|
_ = source_jobs.next() => {
|
||||||
|
warn!("shutting down quic_plugin_source because subtask failed...");
|
||||||
|
break;
|
||||||
|
},
|
||||||
|
_ = exit.recv() => {
|
||||||
|
warn!("shutting down quic_plugin_source...");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
msg = msg_receiver.recv() => {
|
||||||
|
match msg {
|
||||||
|
Ok(msg) => {
|
||||||
|
process_account_updated_from_sources(&account_write_queue_sender,
|
||||||
|
&slot_queue_sender,
|
||||||
|
&msg_receiver,
|
||||||
|
msg,
|
||||||
|
&mut latest_write,
|
||||||
|
latest_write_retention,
|
||||||
|
&metdata_write_queue_sender,
|
||||||
|
&filters,
|
||||||
|
).await ;
|
||||||
|
}
|
||||||
|
Err(e) => {
|
||||||
|
warn!("failed to process quic event: {:?}", e);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
// close all channels to notify downstream CSPs of error
|
||||||
|
account_write_queue_sender.close();
|
||||||
|
metdata_write_queue_sender.map(|s| s.close());
|
||||||
|
slot_queue_sender.close();
|
||||||
|
}
|
||||||
|
|
||||||
|
// consume channel with snapshot and update data
|
||||||
|
async fn process_account_updated_from_sources(
|
||||||
|
account_write_queue_sender: &Sender<AccountOrSnapshotUpdate>,
|
||||||
|
slot_queue_sender: &Sender<SlotUpdate>,
|
||||||
|
msg_receiver: &Receiver<SourceMessage>,
|
||||||
|
msg: SourceMessage,
|
||||||
|
latest_write: &mut HashMap<Slot, HashMap<Pubkey, u64>>,
|
||||||
|
// in slots
|
||||||
|
latest_write_retention: u64,
|
||||||
|
// metric_account_writes: &mut MetricU64,
|
||||||
|
// metric_account_queue: &mut MetricU64,
|
||||||
|
// metric_dedup_queue: &mut MetricU64,
|
||||||
|
// metric_slot_queue: &mut MetricU64,
|
||||||
|
// metric_slot_updates: &mut MetricU64,
|
||||||
|
// metric_snapshots: &mut MetricU64,
|
||||||
|
// metric_snapshot_account_writes: &mut MetricU64,
|
||||||
|
metdata_write_queue_sender: &Option<Sender<FeedMetadata>>,
|
||||||
|
filters: &HashSet<Pubkey>,
|
||||||
|
) {
|
||||||
|
let metadata_sender = |msg| {
|
||||||
|
if let Some(sender) = &metdata_write_queue_sender {
|
||||||
|
sender.send_blocking(msg)
|
||||||
|
} else {
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
metrics::QUIC_DEDUP_QUEUE.set(msg_receiver.len() as i64);
|
||||||
|
match msg {
|
||||||
|
SourceMessage::QuicMessage(message) => {
|
||||||
|
match message {
|
||||||
|
Message::AccountMsg(account_message) => {
|
||||||
|
metrics::QUIC_ACCOUNT_WRITES.inc();
|
||||||
|
metrics::QUIC_ACCOUNT_WRITE_QUEUE.set(account_write_queue_sender.len() as i64);
|
||||||
|
let solana_account = account_message.solana_account();
|
||||||
|
|
||||||
|
// Skip writes that a different server has already sent
|
||||||
|
let pubkey_writes = latest_write
|
||||||
|
.entry(account_message.slot_identifier.slot)
|
||||||
|
.or_default();
|
||||||
|
if !filters.contains(&account_message.pubkey) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
let writes = pubkey_writes.entry(account_message.pubkey).or_insert(0);
|
||||||
|
if account_message.write_version <= *writes {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
*writes = account_message.write_version;
|
||||||
|
latest_write.retain(|&k, _| {
|
||||||
|
k >= account_message.slot_identifier.slot - latest_write_retention
|
||||||
|
});
|
||||||
|
|
||||||
|
account_write_queue_sender
|
||||||
|
.send(AccountOrSnapshotUpdate::AccountUpdate(AccountWrite {
|
||||||
|
pubkey: account_message.pubkey,
|
||||||
|
slot: account_message.slot_identifier.slot,
|
||||||
|
write_version: account_message.write_version,
|
||||||
|
lamports: account_message.lamports,
|
||||||
|
owner: account_message.owner,
|
||||||
|
executable: account_message.executable,
|
||||||
|
rent_epoch: account_message.rent_epoch,
|
||||||
|
data: solana_account.data,
|
||||||
|
}))
|
||||||
|
.await
|
||||||
|
.expect("send success");
|
||||||
|
}
|
||||||
|
Message::SlotMsg(slot_message) => {
|
||||||
|
metrics::QUIC_SLOT_UPDATES.inc();
|
||||||
|
metrics::QUIC_SLOT_UPDATE_QUEUE.set(slot_queue_sender.len() as i64);
|
||||||
|
|
||||||
|
let status = if slot_message.commitment_config.is_processed() {
|
||||||
|
SlotStatus::Processed
|
||||||
|
} else if slot_message.commitment_config.is_confirmed() {
|
||||||
|
SlotStatus::Confirmed
|
||||||
|
} else {
|
||||||
|
SlotStatus::Rooted
|
||||||
|
};
|
||||||
|
|
||||||
|
let slot_update = SlotUpdate {
|
||||||
|
slot: slot_message.slot,
|
||||||
|
parent: Some(slot_message.parent),
|
||||||
|
status,
|
||||||
|
};
|
||||||
|
|
||||||
|
slot_queue_sender
|
||||||
|
.send(slot_update)
|
||||||
|
.await
|
||||||
|
.expect("send success");
|
||||||
|
}
|
||||||
|
_ => {
|
||||||
|
// ignore update
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
SourceMessage::Snapshot(update) => {
|
||||||
|
let label = if let Some(prg) = update.program_id {
|
||||||
|
if prg == spl_token::ID {
|
||||||
|
"gpa(tokens)"
|
||||||
|
} else {
|
||||||
|
"gpa"
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
"gma"
|
||||||
|
};
|
||||||
|
metrics::ACCOUNT_SNAPSHOTS
|
||||||
|
.with_label_values(&[&label])
|
||||||
|
.inc();
|
||||||
|
debug!(
|
||||||
|
"processing snapshot for program_id {} -> size={} & missing size={}...",
|
||||||
|
update
|
||||||
|
.program_id
|
||||||
|
.map(|x| x.to_string())
|
||||||
|
.unwrap_or("".to_string()),
|
||||||
|
update.accounts.len(),
|
||||||
|
update.missing_accounts.len()
|
||||||
|
);
|
||||||
|
if let Err(e) = metadata_sender(FeedMetadata::SnapshotStart(update.program_id)) {
|
||||||
|
warn!("failed to send feed matadata event: {}", e);
|
||||||
|
}
|
||||||
|
|
||||||
|
let mut updated_accounts = vec![];
|
||||||
|
for account in update.accounts {
|
||||||
|
metrics::QUIC_SNAPSHOT_ACCOUNT_WRITES.inc();
|
||||||
|
metrics::QUIC_ACCOUNT_WRITE_QUEUE.set(account_write_queue_sender.len() as i64);
|
||||||
|
|
||||||
|
if !filters.contains(&account.pubkey) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
updated_accounts.push(account);
|
||||||
|
}
|
||||||
|
account_write_queue_sender
|
||||||
|
.send(AccountOrSnapshotUpdate::SnapshotUpdate(updated_accounts))
|
||||||
|
.await
|
||||||
|
.expect("send success");
|
||||||
|
|
||||||
|
for account in update.missing_accounts {
|
||||||
|
if let Err(e) = metadata_sender(FeedMetadata::InvalidAccount(account)) {
|
||||||
|
warn!("failed to send feed matadata event: {}", e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
debug!("processing snapshot done");
|
||||||
|
if let Err(e) = metadata_sender(FeedMetadata::SnapshotEnd(update.program_id)) {
|
||||||
|
warn!("failed to send feed matadata event: {}", e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,5 +1,3 @@
|
||||||
use std::any::Any;
|
|
||||||
use std::panic;
|
|
||||||
use anchor_lang::Id;
|
use anchor_lang::Id;
|
||||||
use anchor_spl::token::Token;
|
use anchor_spl::token::Token;
|
||||||
use anchor_spl::token_2022::spl_token_2022::extension::transfer_fee::TransferFeeConfig;
|
use anchor_spl::token_2022::spl_token_2022::extension::transfer_fee::TransferFeeConfig;
|
||||||
|
@ -14,6 +12,8 @@ use solana_program::clock::Clock;
|
||||||
use solana_program::pubkey::Pubkey;
|
use solana_program::pubkey::Pubkey;
|
||||||
use solana_program::sysvar::Sysvar;
|
use solana_program::sysvar::Sysvar;
|
||||||
use solana_sdk::account::ReadableAccount;
|
use solana_sdk::account::ReadableAccount;
|
||||||
|
use std::any::Any;
|
||||||
|
use std::panic;
|
||||||
|
|
||||||
use router_lib::dex::{DexEdge, DexEdgeIdentifier};
|
use router_lib::dex::{DexEdge, DexEdgeIdentifier};
|
||||||
|
|
||||||
|
@ -189,8 +189,7 @@ pub fn swap_base_output(
|
||||||
output_mint: &Option<TransferFeeConfig>,
|
output_mint: &Option<TransferFeeConfig>,
|
||||||
amount_out: u64,
|
amount_out: u64,
|
||||||
) -> anyhow::Result<(u64, u64, u64)> {
|
) -> anyhow::Result<(u64, u64, u64)> {
|
||||||
let res = panic::catch_unwind(||
|
let res = panic::catch_unwind(|| {
|
||||||
{
|
|
||||||
let pool_state = pool;
|
let pool_state = pool;
|
||||||
let block_timestamp = pool_state.open_time + 1; // TODO FAS his is suppose to be the clock
|
let block_timestamp = pool_state.open_time + 1; // TODO FAS his is suppose to be the clock
|
||||||
if !pool_state.get_status_by_bit(PoolStatusBitIndex::Swap)
|
if !pool_state.get_status_by_bit(PoolStatusBitIndex::Swap)
|
||||||
|
@ -269,7 +268,6 @@ pub fn swap_base_output(
|
||||||
} else {
|
} else {
|
||||||
anyhow::bail!("Something went wrong in raydium cp")
|
anyhow::bail!("Something went wrong in raydium cp")
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn get_transfer_fee(
|
pub fn get_transfer_fee(
|
||||||
|
|
|
@ -11,6 +11,14 @@ pub struct GrpcSourceConfig {
|
||||||
pub tls: Option<TlsConfig>,
|
pub tls: Option<TlsConfig>,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(Clone, Debug, Default, serde_derive::Deserialize)]
|
||||||
|
pub struct QuicSourceConfig {
|
||||||
|
pub name: String,
|
||||||
|
pub connection_string: String,
|
||||||
|
pub retry_connection_sleep_secs: u64,
|
||||||
|
pub enable_gso: Option<bool>,
|
||||||
|
}
|
||||||
|
|
||||||
#[derive(Clone, Debug, Default, serde_derive::Deserialize)]
|
#[derive(Clone, Debug, Default, serde_derive::Deserialize)]
|
||||||
pub struct TlsConfig {
|
pub struct TlsConfig {
|
||||||
pub ca_cert_path: String,
|
pub ca_cert_path: String,
|
||||||
|
@ -36,6 +44,7 @@ pub struct Config {
|
||||||
pub safety_checks: Option<SafetyCheckConfig>,
|
pub safety_checks: Option<SafetyCheckConfig>,
|
||||||
pub hot_mints: Option<HotMintsConfig>,
|
pub hot_mints: Option<HotMintsConfig>,
|
||||||
pub debug_config: Option<DebugConfig>,
|
pub debug_config: Option<DebugConfig>,
|
||||||
|
pub snapshot_timeout_in_seconds: Option<u64>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Config {
|
impl Config {
|
||||||
|
@ -78,16 +87,15 @@ pub struct InfinityConfig {
|
||||||
#[derive(Clone, Debug, Default, serde_derive::Deserialize)]
|
#[derive(Clone, Debug, Default, serde_derive::Deserialize)]
|
||||||
pub struct AccountDataSourceConfig {
|
pub struct AccountDataSourceConfig {
|
||||||
pub region: Option<String>,
|
pub region: Option<String>,
|
||||||
pub use_quic: Option<bool>,
|
pub quic_sources: Option<Vec<QuicSourceConfig>>,
|
||||||
#[serde(deserialize_with = "serde_opt_string_or_env", default)]
|
|
||||||
pub quic_address: Option<String>,
|
|
||||||
#[serde(deserialize_with = "serde_string_or_env")]
|
#[serde(deserialize_with = "serde_string_or_env")]
|
||||||
pub rpc_http_url: String,
|
pub rpc_http_url: String,
|
||||||
// does RPC node support getProgramAccountsCompressed
|
// does RPC node support getProgramAccountsCompressed
|
||||||
pub rpc_support_compression: Option<bool>,
|
pub rpc_support_compression: Option<bool>,
|
||||||
pub re_snapshot_interval_secs: Option<u64>,
|
pub re_snapshot_interval_secs: Option<u64>,
|
||||||
pub grpc_sources: Vec<GrpcSourceConfig>,
|
pub grpc_sources: Option<Vec<GrpcSourceConfig>>,
|
||||||
pub dedup_queue_size: usize,
|
pub dedup_queue_size: usize,
|
||||||
|
pub request_timeout_in_seconds: Option<u64>,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Clone, Debug, serde_derive::Deserialize)]
|
#[derive(Clone, Debug, serde_derive::Deserialize)]
|
||||||
|
|
|
@ -185,7 +185,11 @@ pub async fn process_tx_events(
|
||||||
let (msg_sender, msg_receiver) = async_channel::bounded::<ExecTx>(config.dedup_queue_size);
|
let (msg_sender, msg_receiver) = async_channel::bounded::<ExecTx>(config.dedup_queue_size);
|
||||||
let mut source_jobs = vec![];
|
let mut source_jobs = vec![];
|
||||||
|
|
||||||
for grpc_source in config.grpc_sources.clone() {
|
let Some(grpc_sources) = config.grpc_sources.clone() else {
|
||||||
|
panic!("There should be atleast one grpc source specified for grpc tx watcher");
|
||||||
|
};
|
||||||
|
|
||||||
|
for grpc_source in grpc_sources.clone() {
|
||||||
let msg_sender = msg_sender.clone();
|
let msg_sender = msg_sender.clone();
|
||||||
|
|
||||||
// Make TLS config if configured
|
// Make TLS config if configured
|
||||||
|
|
|
@ -208,8 +208,7 @@ async fn run_all_swap_from_dump(dump_name: &str) -> Result<Result<(), Error>, Er
|
||||||
quote.output_amount != received_out_amount
|
quote.output_amount != received_out_amount
|
||||||
};
|
};
|
||||||
|
|
||||||
if unexpected_in_amount || unexpected_out_amount
|
if unexpected_in_amount || unexpected_out_amount {
|
||||||
{
|
|
||||||
debug_print_ix(
|
debug_print_ix(
|
||||||
&mut success,
|
&mut success,
|
||||||
&mut index,
|
&mut index,
|
||||||
|
@ -270,7 +269,11 @@ async fn debug_print_ix(
|
||||||
error!(
|
error!(
|
||||||
"Faulty swapping #{} quote{}: \r\n{} -> {} ({} -> {})\r\n (successfully run {} swap)",
|
"Faulty swapping #{} quote{}: \r\n{} -> {} ({} -> {})\r\n (successfully run {} swap)",
|
||||||
index,
|
index,
|
||||||
if quote.is_exact_out { " (ExactOut)" } else { "" },
|
if quote.is_exact_out {
|
||||||
|
" (ExactOut)"
|
||||||
|
} else {
|
||||||
|
""
|
||||||
|
},
|
||||||
quote.input_mint,
|
quote.input_mint,
|
||||||
quote.output_mint,
|
quote.output_mint,
|
||||||
quote.input_amount,
|
quote.input_amount,
|
||||||
|
|
Loading…
Reference in New Issue