-
Notifications
You must be signed in to change notification settings - Fork 16
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat(db): use
insertMany
for initial unspent outputs (#566)
* Use insert many for initial unspent outputs. * Separate output and ledger update inserts. Upsert outputs until we are certain we can batch insert them. * fmt * Use unordered inserts * Remove transactions and tune read unspent output performance * Always insert many * Separate OutputWithMetadata. Use trystreamext for read unspent outputs. Move ledger update stream. Cleanup. * docs * Minor cleanup and suggestions for #566 (#587) * Increase number of retries to work with large snapshots * Remove `replSet`s from Docker * Refactor * Use tasks again to parallelize. Refactor helper fns to avoid inconsistencies. Co-authored-by: Alexandcoats <alexandcoats@gmail.com> Co-authored-by: Jochen Görtler <grtlr@users.noreply.github.com>
- Loading branch information
Showing
14 changed files
with
505 additions
and
377 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Large diffs are not rendered by default.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,118 @@ | ||
// Copyright 2022 IOTA Stiftung | ||
// SPDX-License-Identifier: Apache-2.0 | ||
|
||
use chronicle::types::ledger::{LedgerOutput, LedgerSpent}; | ||
use futures::Stream; | ||
use pin_project::pin_project; | ||
|
||
use super::{InxError, LedgerUpdateRecord}; | ||
|
||
#[pin_project] | ||
pub struct LedgerUpdateStream<S> { | ||
#[pin] | ||
inner: S, | ||
#[pin] | ||
record: Option<LedgerUpdateRecord>, | ||
} | ||
|
||
impl<S> LedgerUpdateStream<S> { | ||
pub fn new(inner: S) -> Self { | ||
Self { inner, record: None } | ||
} | ||
} | ||
|
||
impl<S: Stream<Item = Result<bee_inx::LedgerUpdate, bee_inx::Error>>> Stream for LedgerUpdateStream<S> { | ||
type Item = Result<LedgerUpdateRecord, InxError>; | ||
|
||
fn poll_next( | ||
self: std::pin::Pin<&mut Self>, | ||
cx: &mut std::task::Context<'_>, | ||
) -> std::task::Poll<Option<Self::Item>> { | ||
use std::task::Poll; | ||
|
||
use bee_inx::LedgerUpdate; | ||
|
||
let mut this = self.project(); | ||
Poll::Ready(loop { | ||
if let Poll::Ready(next) = this.inner.as_mut().poll_next(cx) { | ||
if let Some(res) = next { | ||
match res { | ||
Ok(ledger_update) => match ledger_update { | ||
LedgerUpdate::Begin(marker) => { | ||
// We shouldn't already have a record. If we do, that's bad. | ||
if let Some(record) = this.record.as_mut().take() { | ||
break Some(Err(InxError::InvalidLedgerUpdateCount { | ||
received: record.consumed.len() + record.created.len(), | ||
expected: record.consumed.capacity() + record.created.capacity(), | ||
})); | ||
} else { | ||
this.record.set(Some(LedgerUpdateRecord { | ||
milestone_index: marker.milestone_index.into(), | ||
created: Vec::with_capacity(marker.created_count), | ||
consumed: Vec::with_capacity(marker.consumed_count), | ||
})); | ||
} | ||
} | ||
LedgerUpdate::Consumed(consumed) => { | ||
if let Some(mut record) = this.record.as_mut().as_pin_mut() { | ||
match LedgerSpent::try_from(consumed) { | ||
Ok(consumed) => { | ||
record.consumed.push(consumed); | ||
} | ||
Err(e) => { | ||
break Some(Err(e.into())); | ||
} | ||
} | ||
} else { | ||
break Some(Err(InxError::InvalidMilestoneState)); | ||
} | ||
} | ||
LedgerUpdate::Created(created) => { | ||
if let Some(mut record) = this.record.as_mut().as_pin_mut() { | ||
match LedgerOutput::try_from(created) { | ||
Ok(created) => { | ||
record.created.push(created); | ||
} | ||
Err(e) => { | ||
break Some(Err(e.into())); | ||
} | ||
} | ||
} else { | ||
break Some(Err(InxError::InvalidMilestoneState)); | ||
} | ||
} | ||
LedgerUpdate::End(marker) => { | ||
if let Some(record) = this.record.as_mut().take() { | ||
if record.created.len() != marker.created_count | ||
|| record.consumed.len() != marker.consumed_count | ||
{ | ||
break Some(Err(InxError::InvalidLedgerUpdateCount { | ||
received: record.consumed.len() + record.created.len(), | ||
expected: marker.consumed_count + marker.created_count, | ||
})); | ||
} | ||
break Some(Ok(record)); | ||
} else { | ||
break Some(Err(InxError::InvalidMilestoneState)); | ||
} | ||
} | ||
}, | ||
Err(e) => { | ||
break Some(Err(e.into())); | ||
} | ||
} | ||
} else { | ||
// If we were supposed to be in the middle of a milestone, something went wrong. | ||
if let Some(record) = this.record.as_mut().take() { | ||
break Some(Err(InxError::InvalidLedgerUpdateCount { | ||
received: record.consumed.len() + record.created.len(), | ||
expected: record.consumed.capacity() + record.created.capacity(), | ||
})); | ||
} else { | ||
break None; | ||
} | ||
} | ||
} | ||
}) | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.