140 lines
4.5 KiB
Rust
Raw Normal View History

2026-01-15 15:44:48 +02:00
use std::sync::Arc;
2026-01-12 15:51:24 +02:00
use anyhow::Result;
use bedrock_client::{BasicAuthCredentials, BedrockClient};
2026-01-15 15:44:48 +02:00
use common::block::HashableBlockData;
2026-01-19 13:55:31 +02:00
use futures::{StreamExt, TryFutureExt};
use log::{info, warn};
2026-01-12 15:51:24 +02:00
use nomos_core::mantle::{
Op, SignedMantleTx,
ops::channel::{ChannelId, inscribe::InscriptionOp},
};
2026-01-15 15:44:48 +02:00
use tokio::sync::{RwLock, mpsc::Sender};
2026-01-16 16:15:21 +02:00
use tokio_retry::Retry;
2026-01-09 15:10:38 +02:00
use url::Url;
2026-01-15 15:44:48 +02:00
use crate::{config::IndexerConfig, message::IndexerToSequencerMessage, state::IndexerState};
2026-01-12 15:51:24 +02:00
pub mod config;
2026-01-15 15:44:48 +02:00
pub mod message;
2026-01-13 15:11:51 +02:00
pub mod state;
2026-01-12 15:51:24 +02:00
2026-01-09 15:10:38 +02:00
pub struct IndexerCore {
2026-01-12 15:51:24 +02:00
pub bedrock_client: BedrockClient,
2026-01-15 15:44:48 +02:00
pub channel_sender: Sender<IndexerToSequencerMessage>,
2026-01-12 15:51:24 +02:00
pub config: IndexerConfig,
2026-01-15 15:44:48 +02:00
pub bedrock_url: Url,
pub channel_id: ChannelId,
2026-01-13 15:11:51 +02:00
pub state: IndexerState,
2026-01-09 15:10:38 +02:00
}
impl IndexerCore {
2026-01-12 15:51:24 +02:00
pub fn new(
addr: &str,
auth: Option<BasicAuthCredentials>,
2026-01-15 15:44:48 +02:00
sender: Sender<IndexerToSequencerMessage>,
2026-01-12 15:51:24 +02:00
config: IndexerConfig,
2026-01-15 15:44:48 +02:00
channel_id: ChannelId,
2026-01-12 15:51:24 +02:00
) -> Result<Self> {
Ok(Self {
bedrock_client: BedrockClient::new(auth)?,
bedrock_url: Url::parse(addr)?,
channel_sender: sender,
config,
2026-01-15 15:44:48 +02:00
channel_id,
2026-01-13 15:11:51 +02:00
// No state setup for now, future task.
state: IndexerState {
2026-01-15 15:44:48 +02:00
latest_seen_block: Arc::new(RwLock::new(0)),
2026-01-13 15:11:51 +02:00
},
2026-01-12 15:51:24 +02:00
})
2026-01-09 15:10:38 +02:00
}
2026-01-12 15:51:24 +02:00
pub async fn subscribe_parse_block_stream(&self) -> Result<()> {
2026-01-16 16:15:21 +02:00
loop {
let mut stream_pinned = Box::pin(
self.bedrock_client
.0
.get_lib_stream(self.bedrock_url.clone())
.await?,
);
info!("Block stream joined");
while let Some(block_info) = stream_pinned.next().await {
let header_id = block_info.header_id;
info!("Observed L1 block at height {}", block_info.height);
// Simple retry strategy on requests
let strategy =
tokio_retry::strategy::FibonacciBackoff::from_millis(self.config.start_delay)
.take(self.config.limit_retry);
if let Some(l1_block) = Retry::spawn(strategy, || {
self.bedrock_client
.0
.get_block_by_id(self.bedrock_url.clone(), header_id)
2026-01-19 13:55:31 +02:00
.inspect_err(|err| warn!("Block fetching failed with err: {err:#?}"))
2026-01-16 16:15:21 +02:00
})
2026-01-12 15:51:24 +02:00
.await?
2026-01-16 16:15:21 +02:00
{
info!("Extracted L1 block at height {}", block_info.height);
let l2_blocks_parsed =
parse_blocks(l1_block.into_transactions().into_iter(), &self.channel_id);
for l2_block in l2_blocks_parsed {
// State modification, will be updated in future
{
let mut guard = self.state.latest_seen_block.write().await;
if l2_block.block_id > *guard {
*guard = l2_block.block_id;
}
2026-01-15 15:44:48 +02:00
}
2026-01-16 16:15:21 +02:00
// Sending data into sequencer, may need to be expanded.
let message = IndexerToSequencerMessage::BlockObserved {
l1_block_id: block_info.height,
l2_block_height: l2_block.block_id,
};
2026-01-15 15:44:48 +02:00
2026-01-16 16:15:21 +02:00
self.channel_sender.send(message.clone()).await?;
2026-01-15 15:44:48 +02:00
2026-01-16 16:15:21 +02:00
info!("Sent message {:#?} to sequencer", message);
}
2026-01-12 15:51:24 +02:00
}
}
2026-01-16 16:15:21 +02:00
// Refetch stream after delay
tokio::time::sleep(std::time::Duration::from_millis(
self.config.resubscribe_interval,
))
.await;
}
2026-01-12 15:51:24 +02:00
}
}
pub fn parse_blocks(
block_txs: impl Iterator<Item = SignedMantleTx>,
decoded_channel_id: &ChannelId,
) -> Vec<HashableBlockData> {
block_txs
.flat_map(|tx| {
tx.mantle_tx
.ops
.iter()
.filter_map(|op| match op {
Op::ChannelInscribe(InscriptionOp {
channel_id,
inscription,
..
}) if channel_id == decoded_channel_id => {
borsh::from_slice::<HashableBlockData>(inscription).ok()
}
_ => None,
})
.collect::<Vec<_>>()
})
.collect()
}