Skip to content

Commit

Permalink
feat(starknet_l1_provider): add SoftDeleteIndexMap
Browse files Browse the repository at this point in the history
Previous implementation inside transaction manager didn't account for
validate, the current implementation is more general, and can in the
future perhaps be reused elsewhere after some changes are made.
  • Loading branch information
Gilad Chase committed Jan 9, 2025
1 parent be0cec0 commit 0c7d6df
Show file tree
Hide file tree
Showing 7 changed files with 186 additions and 46 deletions.
2 changes: 1 addition & 1 deletion crates/starknet_l1_provider/src/l1_provider.rs
Original file line number Diff line number Diff line change
Expand Up @@ -49,7 +49,7 @@ impl L1Provider {
) -> L1ProviderResult<ValidationStatus> {
self.validate_height(height)?;
match self.state {
ProviderState::Validate => Ok(self.tx_manager.tx_status(tx_hash)),
ProviderState::Validate => Ok(self.tx_manager.validate_tx(tx_hash)),
ProviderState::Propose => Err(L1ProviderError::ValidateTransactionConsensusBug),
ProviderState::Pending => Err(L1ProviderError::ValidateInPendingState),
ProviderState::Uninitialized => panic!("Uninitialized L1 provider"),
Expand Down
2 changes: 1 addition & 1 deletion crates/starknet_l1_provider/src/l1_provider_tests.rs
Original file line number Diff line number Diff line change
Expand Up @@ -62,7 +62,7 @@ fn validate_happy_flow() {
// Transaction wasn't deleted after the validation.
assert_eq!(
l1_provider.validate(tx_hash!(1), BlockNumber(1)).unwrap(),
ValidationStatus::Validated
ValidationStatus::AlreadyIncludedInPropsedBlock
);
}

Expand Down
2 changes: 2 additions & 0 deletions crates/starknet_l1_provider/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,8 @@ pub mod l1_provider;
pub mod l1_scraper;
pub(crate) mod transaction_manager;

mod soft_delete_index_map;

#[cfg(test)]
pub mod test_utils;

Expand Down
134 changes: 134 additions & 0 deletions crates/starknet_l1_provider/src/soft_delete_index_map.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,134 @@
use std::collections::HashSet;

use indexmap::map::Entry;
use indexmap::IndexMap;
use starknet_api::executable_transaction::L1HandlerTransaction;
use starknet_api::transaction::TransactionHash;

/// An IndexMap that supports soft deletion of entries.
/// Entries marked as deleted remain hidden in the map, allowing for potential recovery,
/// selective permanent deletion, or rollback before being purged.
// TODO: replace with a fully generic struct if there's a need for it.
// TODO: replace with a BTreeIndexMap if commit performance becomes an issue, see note in commit.
#[derive(Clone, Debug, Default)]
pub struct SoftDeleteIndexMap {
txs: IndexMap<TransactionHash, TransactionEntry>,
staged_txs: HashSet<TransactionHash>,
}

impl SoftDeleteIndexMap {
pub fn _new() -> Self {
Self::default()
}

/// Inserts a transaction into the map, returning the previous transaction if it existed.
pub fn _insert(&mut self, tx: L1HandlerTransaction) -> Option<L1HandlerTransaction> {
match self.txs.entry(tx.tx_hash) {
Entry::Occupied(entry) => {
debug_assert_eq!(entry.get().transaction, tx);
Some(entry.get().transaction.clone())
}
Entry::Vacant(entry) => {
entry.insert(TransactionEntry::new(tx));
None
}
}
}

/// Soft delete and return a reference to the first unstaged transaction, by insertion order.
pub fn soft_pop_front(&mut self) -> Option<&L1HandlerTransaction> {
let entry = self.txs.iter().find(|(_, tx)| tx.is_available());
let (&tx_hash, _) = entry?;
self.soft_remove(tx_hash)
}

/// Stages the given transaction with the given hash if it exists and is not already staged, and
/// returns a reference to it.
pub fn soft_remove(&mut self, tx_hash: TransactionHash) -> Option<&L1HandlerTransaction> {
let entry = self.txs.get_mut(&tx_hash)?;

if !entry.is_available() {
return None;
}

debug_assert_eq!(self.staged_txs.get(&tx_hash), None);
entry.set_state(TxState::Staged);
self.staged_txs.insert(tx_hash);

Some(&entry.transaction)
}

/// Commits given transactions by removing them entirely and returning the removed transactions.
/// Uncommitted staged transactions are rolled back to unstaged first.
// Performance note: This operation is linear time with both the number
// of known transactions and the number of committed transactions. This is assumed to be
// good enough while l1-handler numbers remain low, but if this changes and we need log(n)
// removals (amortized), replace indexmap with this (basically a BTreeIndexMap):
// BTreeMap<u32, TransactionEntry>, Hashmap<TransactionHash, u32> and a counter: u32, such that
// every new tx is inserted to the map with key counter++ and the counter is not reduced
// when removing entries. Once the counter reaches u32::MAX/2 we recreate the DS in Theta(n).
pub fn _commit(&mut self, tx_hashes: &[TransactionHash]) -> Vec<L1HandlerTransaction> {
self._rollback_staging();
let tx_hashes: HashSet<_> = tx_hashes.iter().copied().collect();
if tx_hashes.is_empty() {
return Vec::new();
}

// NOTE: this takes Theta(|self.txs|), see docstring.
let (committed, not_committed): (Vec<_>, Vec<_>) =
self.txs.drain(..).partition(|(hash, _)| tx_hashes.contains(hash));
self.txs.extend(not_committed);

committed.into_iter().map(|(_, entry)| entry.transaction).collect()
}

/// Rolls back all staged transactions, converting them to unstaged.
pub fn _rollback_staging(&mut self) {
for tx_hash in self.staged_txs.drain() {
self.txs.entry(tx_hash).and_modify(|entry| entry.set_state(TxState::Unstaged));
}
}

pub fn is_staged(&self, tx_hash: &TransactionHash) -> bool {
self.staged_txs.contains(tx_hash)
}
}

impl From<Vec<L1HandlerTransaction>> for SoftDeleteIndexMap {
fn from(txs: Vec<L1HandlerTransaction>) -> Self {
let txs = txs.into_iter().map(|tx| (tx.tx_hash, TransactionEntry::new(tx))).collect();
SoftDeleteIndexMap { txs, ..Default::default() }
}
}

/// Indicates whether a transaction is unstaged or staged.
#[derive(Debug, Clone)]
enum TxState {
Unstaged,
Staged,
}

/// Wraps an L1HandlerTransaction along with its current TxState,
/// and provides convenience methods for stage/unstage.
#[derive(Debug, Clone)]
struct TransactionEntry {
pub transaction: L1HandlerTransaction,
pub state: TxState,
}

impl TransactionEntry {
pub fn new(transaction: L1HandlerTransaction) -> Self {
Self { transaction, state: TxState::Unstaged }
}

pub fn set_state(&mut self, state: TxState) {
self.state = state
}

pub fn is_available(&self) -> bool {
match self.state {
TxState::Unstaged => true,
TxState::Staged => false,
}
}
}
53 changes: 26 additions & 27 deletions crates/starknet_l1_provider/src/test_utils.rs
Original file line number Diff line number Diff line change
@@ -1,11 +1,15 @@
use indexmap::{IndexMap, IndexSet};
use std::mem;

use indexmap::IndexSet;
use starknet_api::block::BlockNumber;
use starknet_api::executable_transaction::L1HandlerTransaction;
use starknet_api::transaction::TransactionHash;

use crate::l1_provider::L1Provider;
use crate::soft_delete_index_map::SoftDeleteIndexMap;
use crate::transaction_manager::TransactionManager;
use crate::ProviderState;

// Represents the internal content of the L1 provider for testing.
// Enables customized (and potentially inconsistent) creation for unit testing.
#[derive(Debug, Default)]
Expand All @@ -18,10 +22,7 @@ pub struct L1ProviderContent {
impl From<L1ProviderContent> for L1Provider {
fn from(content: L1ProviderContent) -> L1Provider {
L1Provider {
tx_manager: content
.tx_manager_content
.map(|tm_content| tm_content.complete_to_tx_manager())
.unwrap_or_default(),
tx_manager: content.tx_manager_content.map(Into::into).unwrap_or_default(),
state: content.state.unwrap_or_default(),
current_height: content.current_height,
}
Expand Down Expand Up @@ -53,8 +54,7 @@ impl L1ProviderContentBuilder {
mut self,
tx_hashes: impl IntoIterator<Item = TransactionHash>,
) -> Self {
self.tx_manager_content_builder =
self.tx_manager_content_builder.with_on_l2_awaiting_l1_consumption(tx_hashes);
self.tx_manager_content_builder = self.tx_manager_content_builder.committed(tx_hashes);
self
}

Expand All @@ -75,37 +75,39 @@ impl L1ProviderContentBuilder {
// Enables customized (and potentially inconsistent) creation for unit testing.
#[derive(Debug, Default)]
struct TransactionManagerContent {
txs: Option<IndexMap<TransactionHash, L1HandlerTransaction>>,
on_l2_awaiting_l1_consumption: Option<IndexSet<TransactionHash>>,
txs: Option<Vec<L1HandlerTransaction>>,
committed: Option<IndexSet<TransactionHash>>,
}

impl TransactionManagerContent {
fn complete_to_tx_manager(self) -> TransactionManager {
impl From<TransactionManagerContent> for TransactionManager {
fn from(mut content: TransactionManagerContent) -> TransactionManager {
let txs: Vec<_> = mem::take(&mut content.txs).unwrap();
TransactionManager {
txs: self.txs.unwrap_or_default(),
on_l2_awaiting_l1_consumption: self.on_l2_awaiting_l1_consumption.unwrap_or_default(),
..Default::default()
txs: SoftDeleteIndexMap::from(txs),
committed: content
.committed
.unwrap_or_default()
.into_iter()
.map(|tx_hash| (tx_hash, None))
.collect(),
}
}
}

#[derive(Debug, Default)]
struct TransactionManagerContentBuilder {
txs: Option<IndexMap<TransactionHash, L1HandlerTransaction>>,
on_l2_awaiting_l1_consumption: Option<IndexSet<TransactionHash>>,
txs: Option<Vec<L1HandlerTransaction>>,
committed: Option<IndexSet<TransactionHash>>,
}

impl TransactionManagerContentBuilder {
fn with_txs(mut self, txs: impl IntoIterator<Item = L1HandlerTransaction>) -> Self {
self.txs = Some(txs.into_iter().map(|tx| (tx.tx_hash, tx)).collect());
self.txs = Some(txs.into_iter().collect());
self
}

fn with_on_l2_awaiting_l1_consumption(
mut self,
tx_hashes: impl IntoIterator<Item = TransactionHash>,
) -> Self {
self.on_l2_awaiting_l1_consumption = Some(tx_hashes.into_iter().collect());
fn committed(mut self, tx_hashes: impl IntoIterator<Item = TransactionHash>) -> Self {
self.committed = Some(tx_hashes.into_iter().collect());
self
}

Expand All @@ -114,13 +116,10 @@ impl TransactionManagerContentBuilder {
return None;
}

Some(TransactionManagerContent {
txs: self.txs,
on_l2_awaiting_l1_consumption: self.on_l2_awaiting_l1_consumption,
})
Some(TransactionManagerContent { txs: self.txs, committed: self.committed })
}

fn is_default(&self) -> bool {
self.txs.is_none() && self.on_l2_awaiting_l1_consumption.is_none()
self.txs.is_none() && self.committed.is_none()
}
}
38 changes: 21 additions & 17 deletions crates/starknet_l1_provider/src/transaction_manager.rs
Original file line number Diff line number Diff line change
@@ -1,34 +1,38 @@
use indexmap::{IndexMap, IndexSet};
use indexmap::IndexMap;
use starknet_api::executable_transaction::L1HandlerTransaction;
use starknet_api::transaction::TransactionHash;
use starknet_l1_provider_types::ValidationStatus;

use crate::soft_delete_index_map::SoftDeleteIndexMap;

#[derive(Debug, Default)]
pub struct TransactionManager {
pub txs: IndexMap<TransactionHash, L1HandlerTransaction>,
pub proposed_txs: IndexSet<TransactionHash>,
pub on_l2_awaiting_l1_consumption: IndexSet<TransactionHash>,
pub txs: SoftDeleteIndexMap,
pub committed: IndexMap<TransactionHash, Option<L1HandlerTransaction>>,
}

impl TransactionManager {
pub fn get_txs(&mut self, n_txs: usize) -> Vec<L1HandlerTransaction> {
let (tx_hashes, txs): (Vec<_>, Vec<_>) = self
.txs
.iter()
.skip(self.proposed_txs.len()) // Transactions are proposed FIFO.
.take(n_txs)
.map(|(&hash, tx)| (hash, tx.clone()))
.unzip();

self.proposed_txs.extend(tx_hashes);
let mut txs = Vec::with_capacity(n_txs);

for _ in 0..n_txs {
match self.txs.soft_pop_front().cloned() {
Some(tx) => txs.push(tx),
None => break,
}
}
txs
}

pub fn tx_status(&self, tx_hash: TransactionHash) -> ValidationStatus {
if self.txs.contains_key(&tx_hash) {
pub fn validate_tx(&mut self, tx_hash: TransactionHash) -> ValidationStatus {
if self.committed.contains_key(&tx_hash) {
return ValidationStatus::AlreadyIncludedOnL2;
}

if self.txs.soft_remove(tx_hash).is_some() {
ValidationStatus::Validated
} else if self.on_l2_awaiting_l1_consumption.contains(&tx_hash) {
ValidationStatus::AlreadyIncludedOnL2
} else if self.txs.is_staged(&tx_hash) {
ValidationStatus::AlreadyIncludedInPropsedBlock
} else {
ValidationStatus::ConsumedOnL1OrUnknown
}
Expand Down
1 change: 1 addition & 0 deletions crates/starknet_l1_provider_types/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,7 @@ pub type SharedL1ProviderClient = Arc<dyn L1ProviderClient>;

#[derive(Clone, Copy, Debug, PartialEq, Eq)]
pub enum ValidationStatus {
AlreadyIncludedInPropsedBlock,
AlreadyIncludedOnL2,
ConsumedOnL1OrUnknown,
Validated,
Expand Down

0 comments on commit 0c7d6df

Please sign in to comment.