geyser: reconstruct blocks with zero entries (#245)
This commit is contained in:
parent
f9394dbd23
commit
6590208d4d
|
@ -16,6 +16,14 @@ The minor version will be incremented upon a breaking change and the patch versi
|
||||||
|
|
||||||
### Breaking
|
### Breaking
|
||||||
|
|
||||||
|
## 2023-11-24
|
||||||
|
|
||||||
|
- yellowstone-grpc-geyser-1.11.1+solana.1.17.6
|
||||||
|
|
||||||
|
### Fixes
|
||||||
|
|
||||||
|
- geyser: reconstruct blocks with zero entries ([#245](https://github.com/rpcpool/yellowstone-grpc/pull/245))
|
||||||
|
|
||||||
## 2023-11-21
|
## 2023-11-21
|
||||||
|
|
||||||
- yellowstone-grpc-client-1.12.0+solana.1.17.6
|
- yellowstone-grpc-client-1.12.0+solana.1.17.6
|
||||||
|
|
|
@ -806,7 +806,6 @@ dependencies = [
|
||||||
"anstream",
|
"anstream",
|
||||||
"anstyle",
|
"anstyle",
|
||||||
"clap_lex",
|
"clap_lex",
|
||||||
"once_cell",
|
|
||||||
"strsim",
|
"strsim",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
@ -4995,7 +4994,7 @@ dependencies = [
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "yellowstone-grpc-geyser"
|
name = "yellowstone-grpc-geyser"
|
||||||
version = "1.11.0+solana.1.17.6"
|
version = "1.11.1+solana.1.17.6"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"base64 0.21.4",
|
"base64 0.21.4",
|
||||||
|
|
22
Cargo.toml
22
Cargo.toml
|
@ -3,7 +3,7 @@ resolver = "2"
|
||||||
members = [
|
members = [
|
||||||
"examples/rust", # 1.11.0+solana.1.17.6
|
"examples/rust", # 1.11.0+solana.1.17.6
|
||||||
"yellowstone-grpc-client", # 1.12.0+solana.1.17.6
|
"yellowstone-grpc-client", # 1.12.0+solana.1.17.6
|
||||||
"yellowstone-grpc-geyser", # 1.11.0+solana.1.17.6
|
"yellowstone-grpc-geyser", # 1.11.1+solana.1.17.6
|
||||||
"yellowstone-grpc-proto", # 1.11.0+solana.1.17.6
|
"yellowstone-grpc-proto", # 1.11.0+solana.1.17.6
|
||||||
"yellowstone-grpc-tools", # 1.0.0-rc.7+solana.1.17.6
|
"yellowstone-grpc-tools", # 1.0.0-rc.7+solana.1.17.6
|
||||||
]
|
]
|
||||||
|
@ -20,14 +20,14 @@ keywords = ["solana"]
|
||||||
anyhow = "1.0.62"
|
anyhow = "1.0.62"
|
||||||
async-trait = "0.1.73"
|
async-trait = "0.1.73"
|
||||||
atty = "0.2.14"
|
atty = "0.2.14"
|
||||||
backoff = { version = "0.4.0", features = ["tokio"] }
|
backoff = "0.4.0"
|
||||||
base64 = "0.21.0"
|
base64 = "0.21.0"
|
||||||
bincode = "1.3.3"
|
bincode = "1.3.3"
|
||||||
bs58 = "0.4.0"
|
bs58 = "0.4.0"
|
||||||
bytes = "1.3.0"
|
bytes = "1.3.0"
|
||||||
cargo-lock = "9.0.0"
|
cargo-lock = "9.0.0"
|
||||||
chrono = "0.4.26"
|
chrono = "0.4.26"
|
||||||
clap = { version = "4.3.0", features = ["cargo", "derive"] }
|
clap = "4.3.0"
|
||||||
const-hex = "1.6.2"
|
const-hex = "1.6.2"
|
||||||
crossbeam-channel = "0.5.8"
|
crossbeam-channel = "0.5.8"
|
||||||
env_logger = "0.10.0"
|
env_logger = "0.10.0"
|
||||||
|
@ -37,7 +37,7 @@ google-cloud-googleapis = "0.11.0"
|
||||||
google-cloud-pubsub = "0.21.0"
|
google-cloud-pubsub = "0.21.0"
|
||||||
hex = "0.4.3"
|
hex = "0.4.3"
|
||||||
http = "0.2.8"
|
http = "0.2.8"
|
||||||
hyper = { version = "0.14.27", features = ["server"] }
|
hyper = "0.14.27"
|
||||||
json5 = "0.4.1"
|
json5 = "0.4.1"
|
||||||
lazy_static = "1.4.0"
|
lazy_static = "1.4.0"
|
||||||
log = "0.4.17"
|
log = "0.4.17"
|
||||||
|
@ -45,8 +45,8 @@ maplit = "1.0.2"
|
||||||
prometheus = "0.13.2"
|
prometheus = "0.13.2"
|
||||||
prost = "0.12.1"
|
prost = "0.12.1"
|
||||||
protobuf-src = "1.1.0"
|
protobuf-src = "1.1.0"
|
||||||
rdkafka = { version = "0.34.0", features = ["sasl"] }
|
rdkafka = "0.34.0"
|
||||||
serde = { version = "1.0.145", features = ["derive"] }
|
serde = "1.0.145"
|
||||||
serde_json = "1.0.86"
|
serde_json = "1.0.86"
|
||||||
serde_yaml = "0.9.25"
|
serde_yaml = "0.9.25"
|
||||||
sha2 = "0.10.7"
|
sha2 = "0.10.7"
|
||||||
|
@ -55,16 +55,16 @@ solana-geyser-plugin-interface = "=1.17.6"
|
||||||
solana-logger = "=1.17.6"
|
solana-logger = "=1.17.6"
|
||||||
solana-sdk = "=1.17.6"
|
solana-sdk = "=1.17.6"
|
||||||
solana-transaction-status = "=1.17.6"
|
solana-transaction-status = "=1.17.6"
|
||||||
spl-token-2022 = { version = "0.9.0", features = ["no-entrypoint"] }
|
spl-token-2022 = "0.9.0"
|
||||||
thiserror = "1.0"
|
thiserror = "1.0"
|
||||||
tokio = { version = "1.21.2", features = ["rt-multi-thread", "macros", "time", "fs", "signal"] }
|
tokio = "1.21.2"
|
||||||
tokio-stream = "0.1.11"
|
tokio-stream = "0.1.11"
|
||||||
tonic = { version = "0.10.2", features = ["gzip", "tls", "tls-roots"] }
|
tonic = "0.10.2"
|
||||||
tonic-build = "0.10.2"
|
tonic-build = "0.10.2"
|
||||||
tonic-health = "0.10.2"
|
tonic-health = "0.10.2"
|
||||||
tracing = "0.1.37"
|
tracing = "0.1.37"
|
||||||
tracing-subscriber = { version = "0.3.17", features = ["env-filter"] }
|
tracing-subscriber = "0.3.17"
|
||||||
vergen = { version = "8.2.1", features = ["build", "rustc"] }
|
vergen = "8.2.1"
|
||||||
yellowstone-grpc-client = { path = "yellowstone-grpc-client", version = "=1.12.0+solana.1.17.6" }
|
yellowstone-grpc-client = { path = "yellowstone-grpc-client", version = "=1.12.0+solana.1.17.6" }
|
||||||
yellowstone-grpc-proto = { path = "yellowstone-grpc-proto", version = "=1.11.0+solana.1.17.6" }
|
yellowstone-grpc-proto = { path = "yellowstone-grpc-proto", version = "=1.11.0+solana.1.17.6" }
|
||||||
|
|
||||||
|
|
|
@ -6,6 +6,10 @@ It provides the ability to get slots, blocks, transactions, and account update n
|
||||||
|
|
||||||
For additional documentation, please see: https://docs.triton.one/rpc-pool/grpc-subscriptions
|
For additional documentation, please see: https://docs.triton.one/rpc-pool/grpc-subscriptions
|
||||||
|
|
||||||
|
#### Known bugs
|
||||||
|
|
||||||
|
Block reconstruction inside gRPC plugin based on information provided by BlockMeta, unfortunately number of entries for blocks generated on validators always equal to zero. These blocks always will have zero entries. See issue on GitHub: https://github.com/solana-labs/solana/issues/33823
|
||||||
|
|
||||||
### Validator
|
### Validator
|
||||||
|
|
||||||
Current plugin version (`+solana.1.16.x`) use validator with backported `ReplicaBlockInfoV3` to Geyser interface — https://github.com/solana-labs/solana/pull/33359. As result it's not compatible with original validator from Solana Labs and would not work. You need to compile validator from the source code and can find patched releases in `Triton One` Solana fork: https://github.com/rpcpool/solana-public/tree/v1.16.16-geyser-block-v3.
|
Current plugin version (`+solana.1.16.x`) use validator with backported `ReplicaBlockInfoV3` to Geyser interface — https://github.com/solana-labs/solana/pull/33359. As result it's not compatible with original validator from Solana Labs and would not work. You need to compile validator from the source code and can find patched releases in `Triton One` Solana fork: https://github.com/rpcpool/solana-public/tree/v1.16.16-geyser-block-v3.
|
||||||
|
|
|
@ -14,11 +14,11 @@ name = "client"
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
anyhow = { workspace = true }
|
anyhow = { workspace = true }
|
||||||
backoff = { workspace = true }
|
backoff = { workspace = true, features = ["tokio"] }
|
||||||
bincode = { workspace = true }
|
bincode = { workspace = true }
|
||||||
bs58 = { workspace = true }
|
bs58 = { workspace = true }
|
||||||
chrono = { workspace = true }
|
chrono = { workspace = true }
|
||||||
clap = { workspace = true }
|
clap = { workspace = true, features = ["derive"] }
|
||||||
env_logger = { workspace = true }
|
env_logger = { workspace = true }
|
||||||
futures = { workspace = true }
|
futures = { workspace = true }
|
||||||
hex = { workspace = true }
|
hex = { workspace = true }
|
||||||
|
|
|
@ -15,9 +15,9 @@ bytes = { workspace = true }
|
||||||
futures = { workspace = true }
|
futures = { workspace = true }
|
||||||
http = { workspace = true }
|
http = { workspace = true }
|
||||||
thiserror ={ workspace = true }
|
thiserror ={ workspace = true }
|
||||||
tonic = { workspace = true }
|
tonic = { workspace = true, features = ["tls", "tls-roots"] }
|
||||||
tonic-health = { workspace = true }
|
tonic-health = { workspace = true }
|
||||||
yellowstone-grpc-proto = { workspace = true }
|
yellowstone-grpc-proto = { workspace = true }
|
||||||
|
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
tokio = { workspace = true }
|
tokio = { workspace = true, features = ["rt-multi-thread", "macros"] }
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
[package]
|
[package]
|
||||||
name = "yellowstone-grpc-geyser"
|
name = "yellowstone-grpc-geyser"
|
||||||
version = "1.11.0+solana.1.17.6"
|
version = "1.11.1+solana.1.17.6"
|
||||||
authors = { workspace = true }
|
authors = { workspace = true }
|
||||||
edition = { workspace = true }
|
edition = { workspace = true }
|
||||||
description = "Yellowstone gRPC Geyser Plugin"
|
description = "Yellowstone gRPC Geyser Plugin"
|
||||||
|
@ -21,7 +21,7 @@ anyhow = { workspace = true }
|
||||||
base64 = { workspace = true }
|
base64 = { workspace = true }
|
||||||
bincode = { workspace = true }
|
bincode = { workspace = true }
|
||||||
bs58 = { workspace = true }
|
bs58 = { workspace = true }
|
||||||
clap = { workspace = true }
|
clap = { workspace = true, features = ["derive"] }
|
||||||
crossbeam-channel = { workspace = true }
|
crossbeam-channel = { workspace = true }
|
||||||
futures = { workspace = true }
|
futures = { workspace = true }
|
||||||
hyper = { workspace = true }
|
hyper = { workspace = true }
|
||||||
|
@ -34,10 +34,10 @@ solana-geyser-plugin-interface = { workspace = true }
|
||||||
solana-logger = { workspace = true }
|
solana-logger = { workspace = true }
|
||||||
solana-sdk = { workspace = true }
|
solana-sdk = { workspace = true }
|
||||||
solana-transaction-status = { workspace = true }
|
solana-transaction-status = { workspace = true }
|
||||||
spl-token-2022 = { workspace = true }
|
spl-token-2022 = { workspace = true, features = ["no-entrypoint"] }
|
||||||
tokio = { workspace = true, features = ["fs"] }
|
tokio = { workspace = true, features = ["rt-multi-thread", "macros", "fs"] }
|
||||||
tokio-stream = { workspace = true }
|
tokio-stream = { workspace = true }
|
||||||
tonic = { workspace = true }
|
tonic = { workspace = true, features = ["gzip", "tls", "tls-roots"] }
|
||||||
tonic-health = { workspace = true }
|
tonic-health = { workspace = true }
|
||||||
yellowstone-grpc-proto = { workspace = true }
|
yellowstone-grpc-proto = { workspace = true }
|
||||||
|
|
||||||
|
@ -45,4 +45,4 @@ yellowstone-grpc-proto = { workspace = true }
|
||||||
anyhow = { workspace = true }
|
anyhow = { workspace = true }
|
||||||
cargo-lock = { workspace = true }
|
cargo-lock = { workspace = true }
|
||||||
git-version = { workspace = true }
|
git-version = { workspace = true }
|
||||||
vergen = { workspace = true }
|
vergen = { workspace = true, features = ["build", "rustc"] }
|
||||||
|
|
|
@ -636,6 +636,7 @@ struct SlotMessages {
|
||||||
accounts_dedup: HashMap<Pubkey, (u64, usize)>, // (write_version, message_index)
|
accounts_dedup: HashMap<Pubkey, (u64, usize)>, // (write_version, message_index)
|
||||||
entries: Vec<MessageEntry>,
|
entries: Vec<MessageEntry>,
|
||||||
sealed: bool,
|
sealed: bool,
|
||||||
|
entries_count: usize,
|
||||||
confirmed_at: Option<usize>,
|
confirmed_at: Option<usize>,
|
||||||
finalized_at: Option<usize>,
|
finalized_at: Option<usize>,
|
||||||
}
|
}
|
||||||
|
@ -644,11 +645,19 @@ impl SlotMessages {
|
||||||
pub fn try_seal(&mut self) -> Option<Message> {
|
pub fn try_seal(&mut self) -> Option<Message> {
|
||||||
if !self.sealed {
|
if !self.sealed {
|
||||||
if let Some(block_meta) = &self.block_meta {
|
if let Some(block_meta) = &self.block_meta {
|
||||||
if self.transactions.len() == block_meta.executed_transaction_count as usize
|
let executed_transaction_count = block_meta.executed_transaction_count as usize;
|
||||||
&& self.entries.len() == block_meta.entries_count as usize
|
let entries_count = block_meta.entries_count as usize;
|
||||||
|
|
||||||
|
// Additional check `entries_count == 0` due to bug of zero entries on block produced by validator
|
||||||
|
// See GitHub issue: https://github.com/solana-labs/solana/issues/33823
|
||||||
|
if self.transactions.len() == executed_transaction_count
|
||||||
|
&& (entries_count == 0 || self.entries.len() == entries_count)
|
||||||
{
|
{
|
||||||
let transactions = std::mem::take(&mut self.transactions);
|
let transactions = std::mem::take(&mut self.transactions);
|
||||||
let entries = std::mem::take(&mut self.entries);
|
let mut entries = std::mem::take(&mut self.entries);
|
||||||
|
if entries_count == 0 {
|
||||||
|
entries.clear();
|
||||||
|
}
|
||||||
|
|
||||||
let mut accounts = Vec::with_capacity(self.messages.len());
|
let mut accounts = Vec::with_capacity(self.messages.len());
|
||||||
for item in self.messages.iter().flatten() {
|
for item in self.messages.iter().flatten() {
|
||||||
|
@ -663,6 +672,7 @@ impl SlotMessages {
|
||||||
self.messages.push(Some(message.clone()));
|
self.messages.push(Some(message.clone()));
|
||||||
|
|
||||||
self.sealed = true;
|
self.sealed = true;
|
||||||
|
self.entries_count = entries_count;
|
||||||
return Some(message);
|
return Some(message);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -865,7 +875,7 @@ impl GrpcService {
|
||||||
slot_messages.messages.push(Some(message.clone()));
|
slot_messages.messages.push(Some(message.clone()));
|
||||||
|
|
||||||
// If we already build Block message, new message will be a problem
|
// If we already build Block message, new message will be a problem
|
||||||
if slot_messages.sealed {
|
if slot_messages.sealed && !(matches!(message, Message::Entry(_)) && slot_messages.entries_count == 0) {
|
||||||
prom::update_invalid_blocks(format!("unexpected message {}", message.kind()));
|
prom::update_invalid_blocks(format!("unexpected message {}", message.kind()));
|
||||||
match block_fail_action {
|
match block_fail_action {
|
||||||
ConfigBlockFailAction::Log => {
|
ConfigBlockFailAction::Log => {
|
||||||
|
|
|
@ -22,7 +22,7 @@ required-features = ["kafka"]
|
||||||
anyhow = { workspace = true }
|
anyhow = { workspace = true }
|
||||||
async-trait = { workspace = true }
|
async-trait = { workspace = true }
|
||||||
atty = { workspace = true }
|
atty = { workspace = true }
|
||||||
clap = { workspace = true }
|
clap = { workspace = true, features = ["derive"] }
|
||||||
const-hex = { workspace = true, optional = true }
|
const-hex = { workspace = true, optional = true }
|
||||||
futures = { workspace = true }
|
futures = { workspace = true }
|
||||||
google-cloud-googleapis = { workspace = true, optional = true }
|
google-cloud-googleapis = { workspace = true, optional = true }
|
||||||
|
@ -35,26 +35,26 @@ serde = { workspace = true }
|
||||||
serde_json = { workspace = true }
|
serde_json = { workspace = true }
|
||||||
serde_yaml = { workspace = true }
|
serde_yaml = { workspace = true }
|
||||||
sha2 = { workspace = true, optional = true }
|
sha2 = { workspace = true, optional = true }
|
||||||
tokio = { workspace = true }
|
tokio = { workspace = true, features = ["signal"] }
|
||||||
tokio-stream = { workspace = true }
|
tokio-stream = { workspace = true }
|
||||||
tonic = { workspace = true }
|
tonic = { workspace = true, features = ["gzip"] }
|
||||||
tonic-health = { workspace = true }
|
tonic-health = { workspace = true }
|
||||||
tracing = { workspace = true }
|
tracing = { workspace = true }
|
||||||
tracing-subscriber = { workspace = true }
|
tracing-subscriber = { workspace = true, features = ["env-filter"] }
|
||||||
yellowstone-grpc-client = { workspace = true }
|
yellowstone-grpc-client = { workspace = true }
|
||||||
yellowstone-grpc-proto = { workspace = true }
|
yellowstone-grpc-proto = { workspace = true }
|
||||||
|
|
||||||
[target.'cfg(not(all(target_os = "macos", target_arch = "aarch64")))'.dependencies]
|
[target.'cfg(not(all(target_os = "macos", target_arch = "aarch64")))'.dependencies]
|
||||||
rdkafka = { workspace = true, features = ["ssl"], optional = true }
|
rdkafka = { workspace = true, features = ["sasl", "ssl"], optional = true }
|
||||||
|
|
||||||
[target.'cfg(all(target_os = "macos", target_arch = "aarch64"))'.dependencies]
|
[target.'cfg(all(target_os = "macos", target_arch = "aarch64"))'.dependencies]
|
||||||
rdkafka = { workspace = true, features = ["ssl-vendored"], optional = true }
|
rdkafka = { workspace = true, features = ["sasl", "ssl-vendored"], optional = true }
|
||||||
|
|
||||||
[build-dependencies]
|
[build-dependencies]
|
||||||
anyhow = { workspace = true }
|
anyhow = { workspace = true }
|
||||||
cargo-lock = { workspace = true }
|
cargo-lock = { workspace = true }
|
||||||
git-version = { workspace = true }
|
git-version = { workspace = true }
|
||||||
vergen = { workspace = true }
|
vergen = { workspace = true, features = ["build", "rustc"] }
|
||||||
|
|
||||||
[features]
|
[features]
|
||||||
default = ["google-pubsub", "kafka"]
|
default = ["google-pubsub", "kafka"]
|
||||||
|
|
Loading…
Reference in New Issue