global: snapshot

This commit is contained in:
nym21
2025-10-22 12:36:35 +02:00
parent 8072c4670c
commit 6cd60a064b
20 changed files with 385 additions and 214 deletions

View File

@@ -11,8 +11,10 @@ build = "build.rs"
[dependencies]
bitcoin = { workspace = true }
brk_error = { workspace = true }
brk_rpc = { workspace = true }
brk_structs = { workspace = true }
derive_deref = { workspace = true }
log = { workspace = true }
parking_lot = { workspace = true }
rustc-hash = { workspace = true }

View File

@@ -1,8 +1,10 @@
use std::{path::Path, sync::Arc, thread, time::Duration};
use std::{path::Path, thread, time::Duration};
use brk_error::Result;
use brk_monitor::Mempool;
use brk_rpc::{Auth, Client};
fn main() {
fn main() -> Result<()> {
// Connect to Bitcoin Core
let bitcoin_dir = Path::new(&std::env::var("HOME").unwrap())
.join("Library")
@@ -10,18 +12,14 @@ fn main() {
.join("Bitcoin");
// let bitcoin_dir = Path::new("/Volumes/WD_BLACK/bitcoin");
let rpc = Box::leak(Box::new(
bitcoincore_rpc::Client::new(
"http://localhost:8332",
bitcoincore_rpc::Auth::CookieFile(bitcoin_dir.join(".cookie")),
)
.unwrap(),
));
let client = Client::new(
"http://localhost:8332",
Auth::CookieFile(bitcoin_dir.join(".cookie")),
)?;
let mempool = Arc::new(Mempool::new(rpc));
let mempool = Mempool::new(client);
// Spawn monitoring thread
let mempool_clone = Arc::clone(&mempool);
let mempool_clone = mempool.clone();
thread::spawn(move || {
mempool_clone.start();
});
@@ -34,4 +32,6 @@ fn main() {
let addresses = mempool.get_addresses();
println!("mempool_address_count: {}", addresses.len());
}
// Ok(())
}

View File

@@ -1,21 +1,36 @@
use std::{thread, time::Duration};
use std::{sync::Arc, thread, time::Duration};
use bitcoin::consensus::encode;
use brk_error::Result;
use brk_rpc::Client;
use brk_structs::{AddressBytes, AddressMempoolStats, Transaction, Txid};
use derive_deref::Deref;
use log::error;
use parking_lot::{RwLock, RwLockReadGuard};
use rustc_hash::{FxHashMap, FxHashSet};
const MAX_FETCHES_PER_CYCLE: usize = 10_000;
pub struct Mempool {
///
/// Mempool monitor
///
/// Thread safe and free to clone
///
#[derive(Clone, Deref)]
pub struct Mempool(Arc<MempoolInner>);
impl Mempool {
pub fn new(client: Client) -> Self {
Self(Arc::new(MempoolInner::new(client)))
}
}
pub struct MempoolInner {
client: Client,
txs: RwLock<FxHashMap<Txid, Transaction>>,
addresses: RwLock<FxHashMap<AddressBytes, (AddressMempoolStats, FxHashSet<Txid>)>>,
}
impl Mempool {
impl MempoolInner {
pub fn new(client: Client) -> Self {
Self {
client,
@@ -34,6 +49,7 @@ impl Mempool {
self.addresses.read()
}
/// Start an infinite update loop with a 1 second interval
pub fn start(&self) {
loop {
if let Err(e) = self.update() {
@@ -43,12 +59,11 @@ impl Mempool {
}
}
pub fn update(&self) -> Result<(), Box<dyn std::error::Error>> {
pub fn update(&self) -> Result<()> {
let txids = self
.client
.get_raw_mempool()?
.into_iter()
.map(Txid::from)
.collect::<FxHashSet<_>>();
let new_txs = {
@@ -61,18 +76,12 @@ impl Mempool {
.collect::<Vec<_>>()
}
.into_iter()
.filter_map(|txid| {
self.client
.get_raw_transaction_hex(&bitcoin::Txid::from(&txid), None)
.ok()
.and_then(|hex| encode::deserialize_hex::<bitcoin::Transaction>(&hex).ok())
.map(|tx| Transaction::from_mempool(tx, self.client))
.map(|tx| (txid, tx))
})
.filter_map(|txid| self.client.get_transaction(&txid).ok().map(|tx| (txid, tx)))
.collect::<FxHashMap<_, _>>();
let mut txs = self.txs.write();
let mut addresses = self.addresses.write();
txs.retain(|txid, tx| {
if txids.contains(txid) {
return true;
@@ -98,6 +107,7 @@ impl Mempool {
});
false
});
new_txs.iter().for_each(|(txid, tx)| {
tx.input
.iter()