1use crate::metrics::{AnnouncedTxTypesMetrics, TxTypesCounter};
6use alloy_primitives::{PrimitiveSignature as Signature, TxHash};
7use derive_more::{Deref, DerefMut};
8use reth_eth_wire::{
9 DedupPayload, Eth68TxMetadata, HandleMempoolData, PartiallyValidData, ValidAnnouncementData,
10 MAX_MESSAGE_SIZE,
11};
12use reth_primitives::TxType;
13use std::{fmt, fmt::Display, mem};
14use tracing::trace;
15
16pub const SIGNATURE_DECODED_SIZE_BYTES: usize = mem::size_of::<Signature>();
18
19pub trait ValidateTx68 {
22 fn should_fetch(
27 &self,
28 ty: u8,
29 hash: &TxHash,
30 size: usize,
31 tx_types_counter: &mut TxTypesCounter,
32 ) -> ValidationOutcome;
33
34 fn max_encoded_tx_length(&self, ty: TxType) -> Option<usize>;
38
39 fn strict_max_encoded_tx_length(&self, ty: TxType) -> Option<usize>;
42
43 fn min_encoded_tx_length(&self, ty: TxType) -> Option<usize>;
48
49 fn strict_min_encoded_tx_length(&self, ty: TxType) -> Option<usize>;
52}
53
54#[derive(Debug, Clone, Copy, PartialEq, Eq)]
58pub enum ValidationOutcome {
59 Fetch,
61 Ignore,
63 ReportPeer,
66}
67
68pub trait PartiallyFilterMessage {
71 fn partially_filter_valid_entries<V>(
74 &self,
75 msg: impl DedupPayload<Value = V> + fmt::Debug,
76 ) -> (FilterOutcome, PartiallyValidData<V>) {
77 if msg.is_empty() {
79 trace!(target: "net::tx",
80 msg=?msg,
81 "empty payload"
82 );
83 return (FilterOutcome::ReportPeer, PartiallyValidData::empty_eth66())
84 }
85
86 let original_len = msg.len();
88 let partially_valid_data = msg.dedup();
89
90 (
91 if partially_valid_data.len() == original_len {
92 FilterOutcome::Ok
93 } else {
94 FilterOutcome::ReportPeer
95 },
96 partially_valid_data,
97 )
98 }
99}
100
101pub trait FilterAnnouncement {
106 fn filter_valid_entries_68(
111 &self,
112 msg: PartiallyValidData<Eth68TxMetadata>,
113 ) -> (FilterOutcome, ValidAnnouncementData)
114 where
115 Self: ValidateTx68;
116
117 fn filter_valid_entries_66(
122 &self,
123 msg: PartiallyValidData<Eth68TxMetadata>,
124 ) -> (FilterOutcome, ValidAnnouncementData);
125}
126
127#[derive(Debug, Clone, Copy, PartialEq, Eq)]
131pub enum FilterOutcome {
132 Ok,
134 ReportPeer,
137}
138
139#[derive(Debug, Default, Deref, DerefMut)]
144pub struct MessageFilter<N = EthMessageFilter>(N);
145
146#[derive(Debug, Default)]
148pub struct EthMessageFilter {
149 announced_tx_types_metrics: AnnouncedTxTypesMetrics,
150}
151
152impl Display for EthMessageFilter {
153 fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
154 write!(f, "EthMessageFilter")
155 }
156}
157
158impl PartiallyFilterMessage for EthMessageFilter {}
159
160impl ValidateTx68 for EthMessageFilter {
161 fn should_fetch(
162 &self,
163 ty: u8,
164 hash: &TxHash,
165 size: usize,
166 tx_types_counter: &mut TxTypesCounter,
167 ) -> ValidationOutcome {
168 let tx_type = match TxType::try_from(ty) {
172 Ok(ty) => ty,
173 Err(_) => {
174 trace!(target: "net::eth-wire",
175 ty=ty,
176 size=size,
177 hash=%hash,
178 network=%self,
179 "invalid tx type in eth68 announcement"
180 );
181
182 return ValidationOutcome::ReportPeer
183 }
184 };
185 tx_types_counter.increase_by_tx_type(tx_type);
186
187 if let Some(strict_min_encoded_tx_length) = self.strict_min_encoded_tx_length(tx_type) {
194 if size < strict_min_encoded_tx_length {
195 trace!(target: "net::eth-wire",
196 ty=ty,
197 size=size,
198 hash=%hash,
199 strict_min_encoded_tx_length=strict_min_encoded_tx_length,
200 network=%self,
201 "invalid tx size in eth68 announcement"
202 );
203
204 return ValidationOutcome::Ignore
205 }
206 }
207 if let Some(reasonable_min_encoded_tx_length) = self.min_encoded_tx_length(tx_type) {
208 if size < reasonable_min_encoded_tx_length {
209 trace!(target: "net::eth-wire",
210 ty=ty,
211 size=size,
212 hash=%hash,
213 reasonable_min_encoded_tx_length=reasonable_min_encoded_tx_length,
214 strict_min_encoded_tx_length=self.strict_min_encoded_tx_length(tx_type),
215 network=%self,
216 "tx size in eth68 announcement, is unreasonably small"
217 );
218
219 }
222 }
223 if let Some(reasonable_max_encoded_tx_length) = self.max_encoded_tx_length(tx_type) {
225 if size > reasonable_max_encoded_tx_length {
226 trace!(target: "net::eth-wire",
227 ty=ty,
228 size=size,
229 hash=%hash,
230 reasonable_max_encoded_tx_length=reasonable_max_encoded_tx_length,
231 strict_max_encoded_tx_length=self.strict_max_encoded_tx_length(tx_type),
232 network=%self,
233 "tx size in eth68 announcement, is unreasonably large"
234 );
235
236 }
239 }
240
241 ValidationOutcome::Fetch
242 }
243
244 fn max_encoded_tx_length(&self, ty: TxType) -> Option<usize> {
245 #[allow(unreachable_patterns, clippy::match_same_arms)]
248 match ty {
249 TxType::Legacy | TxType::Eip2930 | TxType::Eip1559 => Some(MAX_MESSAGE_SIZE),
250 TxType::Eip4844 => None,
251 _ => None,
252 }
253 }
254
255 fn strict_max_encoded_tx_length(&self, _ty: TxType) -> Option<usize> {
256 None
257 }
258
259 fn min_encoded_tx_length(&self, _ty: TxType) -> Option<usize> {
260 Some(SIGNATURE_DECODED_SIZE_BYTES)
263 }
264
265 fn strict_min_encoded_tx_length(&self, _ty: TxType) -> Option<usize> {
266 Some(1)
268 }
269}
270
271impl FilterAnnouncement for EthMessageFilter {
272 fn filter_valid_entries_68(
273 &self,
274 mut msg: PartiallyValidData<Eth68TxMetadata>,
275 ) -> (FilterOutcome, ValidAnnouncementData)
276 where
277 Self: ValidateTx68,
278 {
279 trace!(target: "net::tx::validation",
280 msg=?*msg,
281 network=%self,
282 "validating eth68 announcement data.."
283 );
284
285 let mut should_report_peer = false;
286 let mut tx_types_counter = TxTypesCounter::default();
287
288 msg.retain(|hash, metadata| {
294 debug_assert!(
295 metadata.is_some(),
296 "metadata should exist for `%hash` in eth68 announcement passed to `%filter_valid_entries_68`,
297`%hash`: {hash}"
298 );
299
300 let Some((ty, size)) = metadata else {
301 return false
302 };
303
304 match self.should_fetch(*ty, hash, *size, &mut tx_types_counter) {
305 ValidationOutcome::Fetch => true,
306 ValidationOutcome::Ignore => false,
307 ValidationOutcome::ReportPeer => {
308 should_report_peer = true;
309 false
310 }
311 }
312 });
313 self.announced_tx_types_metrics.update_eth68_announcement_metrics(tx_types_counter);
314 (
315 if should_report_peer { FilterOutcome::ReportPeer } else { FilterOutcome::Ok },
316 ValidAnnouncementData::from_partially_valid_data(msg),
317 )
318 }
319
320 fn filter_valid_entries_66(
321 &self,
322 partially_valid_data: PartiallyValidData<Option<(u8, usize)>>,
323 ) -> (FilterOutcome, ValidAnnouncementData) {
324 trace!(target: "net::tx::validation",
325 hashes=?*partially_valid_data,
326 network=%self,
327 "validating eth66 announcement data.."
328 );
329
330 (FilterOutcome::Ok, ValidAnnouncementData::from_partially_valid_data(partially_valid_data))
331 }
332}
333
334#[cfg(test)]
336mod test {
337 use super::*;
338 use alloy_primitives::B256;
339 use reth_eth_wire::{NewPooledTransactionHashes66, NewPooledTransactionHashes68};
340 use std::{collections::HashMap, str::FromStr};
341
342 #[test]
343 fn eth68_empty_announcement() {
344 let types = vec![];
345 let sizes = vec![];
346 let hashes = vec![];
347
348 let announcement = NewPooledTransactionHashes68 { types, sizes, hashes };
349
350 let filter = EthMessageFilter::default();
351
352 let (outcome, _partially_valid_data) = filter.partially_filter_valid_entries(announcement);
353
354 assert_eq!(outcome, FilterOutcome::ReportPeer);
355 }
356
357 #[test]
358 fn eth68_announcement_unrecognized_tx_type() {
359 let types = vec![
360 TxType::MAX_RESERVED_EIP as u8 + 1, TxType::Legacy as u8,
362 ];
363 let sizes = vec![MAX_MESSAGE_SIZE, MAX_MESSAGE_SIZE];
364 let hashes = vec![
365 B256::from_str("0xbeefcafebeefcafebeefcafebeefcafebeefcafebeefcafebeefcafebeefcafa")
366 .unwrap(),
367 B256::from_str("0xbeefcafebeefcafebeefcafebeefcafebeefcafebeefcafebeefcafebeefbbbb")
368 .unwrap(),
369 ];
370
371 let announcement = NewPooledTransactionHashes68 {
372 types: types.clone(),
373 sizes: sizes.clone(),
374 hashes: hashes.clone(),
375 };
376
377 let filter = EthMessageFilter::default();
378
379 let (outcome, partially_valid_data) = filter.partially_filter_valid_entries(announcement);
380
381 assert_eq!(outcome, FilterOutcome::Ok);
382
383 let (outcome, valid_data) = filter.filter_valid_entries_68(partially_valid_data);
384
385 assert_eq!(outcome, FilterOutcome::ReportPeer);
386
387 let mut expected_data = HashMap::default();
388 expected_data.insert(hashes[1], Some((types[1], sizes[1])));
389
390 assert_eq!(expected_data, valid_data.into_data())
391 }
392
393 #[test]
394 fn eth68_announcement_too_small_tx() {
395 let types =
396 vec![TxType::MAX_RESERVED_EIP as u8, TxType::Legacy as u8, TxType::Eip2930 as u8];
397 let sizes = vec![
398 0, 0, 1,
401 ];
402 let hashes = vec![
403 B256::from_str("0xbeefcafebeefcafebeefcafebeefcafebeefcafebeefcafebeefcafebeefcafa")
404 .unwrap(),
405 B256::from_str("0xbeefcafebeefcafebeefcafebeefcafebeefcafebeefcafebeefcafebeefbbbb")
406 .unwrap(),
407 B256::from_str("0xbeefcafebeefcafebeefcafebeefcafebeefcafebeefcafebeefcafebeef00bb")
408 .unwrap(),
409 ];
410
411 let announcement = NewPooledTransactionHashes68 {
412 types: types.clone(),
413 sizes: sizes.clone(),
414 hashes: hashes.clone(),
415 };
416
417 let filter = EthMessageFilter::default();
418
419 let (outcome, partially_valid_data) = filter.partially_filter_valid_entries(announcement);
420
421 assert_eq!(outcome, FilterOutcome::Ok);
422
423 let (outcome, valid_data) = filter.filter_valid_entries_68(partially_valid_data);
424
425 assert_eq!(outcome, FilterOutcome::Ok);
426
427 let mut expected_data = HashMap::default();
428 expected_data.insert(hashes[2], Some((types[2], sizes[2])));
429
430 assert_eq!(expected_data, valid_data.into_data())
431 }
432
433 #[test]
434 fn eth68_announcement_duplicate_tx_hash() {
435 let types = vec![
436 TxType::Eip1559 as u8,
437 TxType::Eip4844 as u8,
438 TxType::Eip1559 as u8,
439 TxType::Eip4844 as u8,
440 ];
441 let sizes = vec![1, 1, 1, MAX_MESSAGE_SIZE];
442 let hashes = vec![
444 B256::from_str("0xbeefcafebeefcafebeefcafebeefcafebeefcafebeefcafebeefcafebeefcafa") .unwrap(),
446 B256::from_str("0xbeefcafebeefcafebeefcafebeefcafebeefcafebeefcafebeefcafebeefcafa") .unwrap(),
448 B256::from_str("0xbeefcafebeefcafebeefcafebeefcafebeefcafebeefcafebeefcafebeefcafa") .unwrap(),
450 B256::from_str("0xbeefcafebeefcafebeefcafebeefcafebeefcafebeefcafebeefcafebeefbbbb")
451 .unwrap(),
452 ];
453
454 let announcement = NewPooledTransactionHashes68 {
455 types: types.clone(),
456 sizes: sizes.clone(),
457 hashes: hashes.clone(),
458 };
459
460 let filter = EthMessageFilter::default();
461
462 let (outcome, partially_valid_data) = filter.partially_filter_valid_entries(announcement);
463
464 assert_eq!(outcome, FilterOutcome::ReportPeer);
465
466 let mut expected_data = HashMap::default();
467 expected_data.insert(hashes[3], Some((types[3], sizes[3])));
468 expected_data.insert(hashes[0], Some((types[0], sizes[0])));
469
470 assert_eq!(expected_data, partially_valid_data.into_data())
471 }
472
473 #[test]
474 fn eth66_empty_announcement() {
475 let hashes = vec![];
476
477 let announcement = NewPooledTransactionHashes66(hashes);
478
479 let filter: MessageFilter = MessageFilter::default();
480
481 let (outcome, _partially_valid_data) = filter.partially_filter_valid_entries(announcement);
482
483 assert_eq!(outcome, FilterOutcome::ReportPeer);
484 }
485
486 #[test]
487 fn eth66_announcement_duplicate_tx_hash() {
488 let hashes = vec![
490 B256::from_str("0xbeefcafebeefcafebeefcafebeefcafebeefcafebeefcafebeefcafebeefbbbb") .unwrap(),
492 B256::from_str("0xbeefcafebeefcafebeefcafebeefcafebeefcafebeefcafebeefcafebeefcafa") .unwrap(),
494 B256::from_str("0xbeefcafebeefcafebeefcafebeefcafebeefcafebeefcafebeefcafebeefcafa") .unwrap(),
496 B256::from_str("0xbeefcafebeefcafebeefcafebeefcafebeefcafebeefcafebeefcafebeefcafa") .unwrap(),
498 B256::from_str("0xbeefcafebeefcafebeefcafebeefcafebeefcafebeefcafebeefcafebeefbbbb") .unwrap(),
500 ];
501
502 let announcement = NewPooledTransactionHashes66(hashes.clone());
503
504 let filter: MessageFilter = MessageFilter::default();
505
506 let (outcome, partially_valid_data) = filter.partially_filter_valid_entries(announcement);
507
508 assert_eq!(outcome, FilterOutcome::ReportPeer);
509
510 let mut expected_data = HashMap::default();
511 expected_data.insert(hashes[1], None);
512 expected_data.insert(hashes[0], None);
513
514 assert_eq!(expected_data, partially_valid_data.into_data())
515 }
516
517 #[test]
518 fn test_display_for_zst() {
519 let filter = EthMessageFilter::default();
520 assert_eq!("EthMessageFilter", &filter.to_string());
521 }
522}