Merge branch 'lmdb_update' into grim
This commit is contained in:
+10
-5
@@ -1293,19 +1293,24 @@ impl Chain {
|
||||
return Ok(());
|
||||
}
|
||||
|
||||
let mut count = 0;
|
||||
let tail_hash = header_pmmr.get_header_hash_by_height(head.height - horizon)?;
|
||||
let tail = batch.get_block_header(&tail_hash)?;
|
||||
|
||||
// Remove old blocks (including short lived fork blocks) which height < tail.height
|
||||
for block in batch.blocks_iter()? {
|
||||
// Remove old blocks (including short-lived fork blocks) which height < tail.height
|
||||
let mut blocks_to_delete = vec![];
|
||||
let iter = batch.blocks_iter()?;
|
||||
for block in iter {
|
||||
if let Ok(block) = block {
|
||||
if block.header.height < tail.height {
|
||||
let _ = batch.delete_block(&block.hash());
|
||||
count += 1;
|
||||
blocks_to_delete.push(block.hash());
|
||||
}
|
||||
}
|
||||
}
|
||||
let mut count = 0;
|
||||
for bh in blocks_to_delete {
|
||||
let _ = batch.delete_block(&bh);
|
||||
count += 1;
|
||||
}
|
||||
|
||||
batch.save_body_tail(&Tip::from_header(&tail))?;
|
||||
|
||||
|
||||
@@ -405,22 +405,36 @@ impl<T: PosEntry> RewindableListIndex for MultiIndex<T> {
|
||||
|
||||
impl<T: PosEntry> PruneableListIndex for MultiIndex<T> {
|
||||
fn clear(&self, batch: &mut Batch<'_>) -> Result<(), Error> {
|
||||
let mut list_count = 0;
|
||||
let mut entry_count = 0;
|
||||
let mut lists_to_delete = vec![];
|
||||
let list_db_key = Some(self.list_prefix);
|
||||
for key in batch.db.iter(list_db_key, |k, _| Ok(k.to_vec()))? {
|
||||
if let Ok(key) = key {
|
||||
let _ = batch.delete(list_db_key, &key);
|
||||
list_count += 1;
|
||||
lists_to_delete.push(key);
|
||||
}
|
||||
}
|
||||
let mut list_count = 0;
|
||||
for l in lists_to_delete {
|
||||
match batch.delete(list_db_key, &l) {
|
||||
Ok(_) => list_count += 1,
|
||||
Err(_) => {}
|
||||
}
|
||||
}
|
||||
|
||||
let mut entries_to_delete = vec![];
|
||||
let entry_db_key = Some(self.entry_prefix);
|
||||
for key in batch.db.iter(entry_db_key, |k, _| Ok(k.to_vec()))? {
|
||||
if let Ok(key) = key {
|
||||
let _ = batch.delete(entry_db_key, &key);
|
||||
entry_count += 1;
|
||||
entries_to_delete.push(key);
|
||||
}
|
||||
}
|
||||
let mut entry_count = 0;
|
||||
for e in entries_to_delete {
|
||||
match batch.delete(entry_db_key, &e) {
|
||||
Ok(_) => entry_count += 1,
|
||||
Err(_) => {}
|
||||
}
|
||||
}
|
||||
|
||||
debug!(
|
||||
"clear: lists deleted: {}, entries deleted: {}",
|
||||
list_count, entry_count
|
||||
|
||||
+5
-5
@@ -340,8 +340,8 @@ impl<'a> Batch<'a> {
|
||||
|
||||
/// Iterator over the output_pos index.
|
||||
pub fn output_pos_iter(
|
||||
&self,
|
||||
) -> Result<impl Iterator<Item = Result<(Vec<u8>, CommitPos), Error>>, Error> {
|
||||
&'a self,
|
||||
) -> Result<impl Iterator<Item = Result<(Vec<u8>, CommitPos), Error>> + 'a, Error> {
|
||||
let protocol_version = self.db.protocol_version();
|
||||
self.db.iter(Some(OUTPUT_POS_PREFIX), move |k, mut v| {
|
||||
ser::deserialize(&mut v, protocol_version, DeserializationMode::default())
|
||||
@@ -461,7 +461,7 @@ impl<'a> Batch<'a> {
|
||||
|
||||
/// Iterator over all full blocks in the db.
|
||||
/// Uses default db serialization strategy via db protocol version.
|
||||
pub fn blocks_iter(&self) -> Result<impl Iterator<Item = Result<Block, Error>> + 'a, Error> {
|
||||
pub fn blocks_iter(&'a self) -> Result<impl Iterator<Item = Result<Block, Error>> + 'a, Error> {
|
||||
let protocol_version = self.db.protocol_version();
|
||||
self.db.iter(Some(BLOCK_PREFIX), move |_, mut v| {
|
||||
ser::deserialize(&mut v, protocol_version, DeserializationMode::default())
|
||||
@@ -472,8 +472,8 @@ impl<'a> Batch<'a> {
|
||||
/// Iterator over raw data for full blocks in the db.
|
||||
/// Used during block migration (we need flexibility around deserialization).
|
||||
pub fn blocks_raw_iter(
|
||||
&self,
|
||||
) -> Result<impl Iterator<Item = Result<(Vec<u8>, Vec<u8>), Error>>, Error> {
|
||||
&'a self,
|
||||
) -> Result<impl Iterator<Item = Result<(Vec<u8>, Vec<u8>), Error>> + 'a, Error> {
|
||||
self.db
|
||||
.iter(Some(BLOCK_PREFIX), |k, v| Ok((k.to_vec(), v.to_vec())))
|
||||
}
|
||||
|
||||
@@ -643,7 +643,7 @@ impl TxHashSet {
|
||||
|
||||
// Iterate over the current output_pos index, removing any entries that
|
||||
// do not point to to the expected output.
|
||||
let mut removed_count = 0;
|
||||
let mut pos_to_delete = vec![];
|
||||
for kp in batch.output_pos_iter()? {
|
||||
if let Ok((key, pos1)) = kp {
|
||||
let pos0 = pos1.pos - 1;
|
||||
@@ -658,10 +658,14 @@ impl TxHashSet {
|
||||
}
|
||||
}
|
||||
}
|
||||
batch.delete(Some(store::OUTPUT_POS_PREFIX), &key)?;
|
||||
removed_count += 1;
|
||||
pos_to_delete.push(key);
|
||||
}
|
||||
}
|
||||
let mut removed_count = 0;
|
||||
for p in pos_to_delete {
|
||||
batch.delete(Some(store::OUTPUT_POS_PREFIX), &p)?;
|
||||
removed_count += 1;
|
||||
}
|
||||
debug!(
|
||||
"init_output_pos_index: removed {} stale index entries",
|
||||
removed_count
|
||||
|
||||
+3
-1
@@ -195,7 +195,9 @@ pub struct PeersIterBatch<'a> {
|
||||
|
||||
impl<'a> PeersIterBatch<'a> {
|
||||
/// Iterator over all known peers.
|
||||
pub fn peers_iter(&self) -> Result<impl Iterator<Item = Result<PeerData, Error>>, Error> {
|
||||
pub fn peers_iter(
|
||||
&'a self,
|
||||
) -> Result<impl Iterator<Item = Result<PeerData, Error>> + 'a, Error> {
|
||||
let protocol_version = self.db.protocol_version();
|
||||
self.db.iter(Some(PEER_PREFIX), move |_, mut v| {
|
||||
ser::deserialize(&mut v, protocol_version, DeserializationMode::default())
|
||||
|
||||
+50
-21
@@ -252,9 +252,15 @@ impl Store {
|
||||
let migrate_from = Path::new(root_path).join(env_name);
|
||||
if migrate_from.exists() {
|
||||
match s.migrate_to_default_env(db_name, &migrate_from) {
|
||||
Ok(_) => {
|
||||
let _ = fs::remove_dir_all(&migrate_from);
|
||||
}
|
||||
Ok(_) => match fs::remove_dir_all(&migrate_from) {
|
||||
Ok(_) => {}
|
||||
Err(e) => {
|
||||
return Err(Error::FileErr(format!(
|
||||
"Can not remove old DB file: {:?}",
|
||||
e
|
||||
)));
|
||||
}
|
||||
},
|
||||
Err(e) => {
|
||||
error!("DB {} migration error: {:?}", env_name, e);
|
||||
match s.clear() {
|
||||
@@ -310,7 +316,7 @@ impl Store {
|
||||
db.put(&mut write_to, key, &v)?;
|
||||
count += 1;
|
||||
} else {
|
||||
error!("Migration: unknown DB key: {}", db_name[0]);
|
||||
warn!("Migration: unknown DB key: {}", db_name[0]);
|
||||
}
|
||||
} else {
|
||||
self.def_db.put(&mut write_to, k, &v)?;
|
||||
@@ -560,11 +566,11 @@ impl Store {
|
||||
}
|
||||
|
||||
/// Produces an iterator from the provided database key.
|
||||
pub fn iter<F, T>(
|
||||
pub fn iter<'a, F, T>(
|
||||
&self,
|
||||
db_key: Option<u8>,
|
||||
deserialize: F,
|
||||
) -> Result<DatabaseIterator<F, T>, Error>
|
||||
) -> Result<DatabaseIterator<'a, F, T>, Error>
|
||||
where
|
||||
F: Fn(&[u8], &[u8]) -> Result<T, Error>,
|
||||
{
|
||||
@@ -662,7 +668,8 @@ impl<'a> Batch<'a> {
|
||||
/// Writes a single key/value pair to the provided database key.
|
||||
pub fn put(&mut self, db_key: Option<u8>, key: &[u8], value: &[u8]) -> Result<(), Error> {
|
||||
let db = self.store.get_db(db_key)?;
|
||||
db.put(&mut self.write, key, value)?;
|
||||
let w = &mut self.write;
|
||||
db.put(w, key, value)?;
|
||||
Ok(())
|
||||
}
|
||||
|
||||
@@ -724,14 +731,38 @@ impl<'a> Batch<'a> {
|
||||
|
||||
/// Produces an iterator from the provided database key.
|
||||
pub fn iter<F, T>(
|
||||
&self,
|
||||
&'a self,
|
||||
db_key: Option<u8>,
|
||||
deserialize: F,
|
||||
) -> Result<DatabaseIterator<F, T>, Error>
|
||||
) -> Result<DatabaseIterator<'a, F, T>, Error>
|
||||
where
|
||||
F: Fn(&[u8], &[u8]) -> Result<T, Error>,
|
||||
{
|
||||
self.store.iter(db_key, deserialize)
|
||||
self.store.wait_for_resize();
|
||||
|
||||
TxCounter::on_change_tx_count(&self.store.env_path, true);
|
||||
let read = self.write.nested_read_txn();
|
||||
match read {
|
||||
Ok(read) => {
|
||||
let db_res = self.store.get_db(db_key);
|
||||
match db_res {
|
||||
Ok(db) => Ok(DatabaseIterator::new(
|
||||
self.store,
|
||||
Arc::new(db.clone()),
|
||||
read,
|
||||
deserialize,
|
||||
)),
|
||||
Err(e) => {
|
||||
TxCounter::on_change_tx_count(&self.store.env_path, false);
|
||||
Err(Error::from(e))
|
||||
}
|
||||
}
|
||||
}
|
||||
Err(e) => {
|
||||
TxCounter::on_change_tx_count(&self.store.env_path, false);
|
||||
Err(Error::from(e))
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/// Gets a `Readable` value from the database by provided key and deserialization strategy.
|
||||
@@ -788,12 +819,12 @@ impl<'a> Batch<'a> {
|
||||
|
||||
/// An iterator based on database key.
|
||||
/// Caller is responsible for deserialization of the data.
|
||||
pub struct DatabaseIterator<F, T>
|
||||
pub struct DatabaseIterator<'a, F, T>
|
||||
where
|
||||
F: Fn(&[u8], &[u8]) -> Result<T, Error>,
|
||||
{
|
||||
db: Arc<Database<Bytes, Bytes>>,
|
||||
read: Arc<RoTxn<'static, WithoutTls>>,
|
||||
read: Arc<RoTxn<'a, WithoutTls>>,
|
||||
keys: Vec<Vec<u8>>,
|
||||
total_keys: usize,
|
||||
skip_cur: usize,
|
||||
@@ -803,7 +834,7 @@ where
|
||||
tx_counter: TxCounter,
|
||||
}
|
||||
|
||||
impl<F, T> Iterator for DatabaseIterator<F, T>
|
||||
impl<F, T> Iterator for DatabaseIterator<'_, F, T>
|
||||
where
|
||||
F: Fn(&[u8], &[u8]) -> Result<T, Error>,
|
||||
{
|
||||
@@ -811,15 +842,13 @@ where
|
||||
|
||||
fn next(&mut self) -> Option<Self::Item> {
|
||||
if let Some(k) = self.keys.iter().skip(self.skip_cur).next() {
|
||||
self.skip_total += 1;
|
||||
self.skip_cur += 1;
|
||||
match self.db.get(&self.read, k) {
|
||||
Ok(v) => {
|
||||
if let Some(v) = v {
|
||||
return match (self.deserialize)(k, v) {
|
||||
Ok(v) => {
|
||||
self.skip_total += 1;
|
||||
self.skip_cur += 1;
|
||||
Some(Ok(v))
|
||||
}
|
||||
Ok(v) => Some(Ok(v)),
|
||||
Err(e) => {
|
||||
error!("db iter: error deserializing: {}", e);
|
||||
Some(Err(Error::from(e)))
|
||||
@@ -853,7 +882,7 @@ where
|
||||
}
|
||||
}
|
||||
|
||||
impl<F, T> DatabaseIterator<F, T>
|
||||
impl<'a, F, T> DatabaseIterator<'a, F, T>
|
||||
where
|
||||
F: Fn(&[u8], &[u8]) -> Result<T, Error>,
|
||||
{
|
||||
@@ -861,9 +890,9 @@ where
|
||||
pub fn new(
|
||||
store: &Store,
|
||||
db: Arc<Database<Bytes, Bytes>>,
|
||||
read: RoTxn<'static, WithoutTls>,
|
||||
read: RoTxn<'a, WithoutTls>,
|
||||
deserialize: F,
|
||||
) -> DatabaseIterator<F, T> {
|
||||
) -> DatabaseIterator<'a, F, T> {
|
||||
let (keys, total_keys) = if let Ok(iter) = db.iter(&read) {
|
||||
let total = iter.move_between_keys().count();
|
||||
let keys = if let Ok(iter) = db.iter(&read) {
|
||||
|
||||
+5
-6
@@ -118,13 +118,12 @@ fn test_iter() -> Result<(), store::Error> {
|
||||
let mut batch = store.batch()?;
|
||||
batch.put(Some(prefix), &key, &value)?;
|
||||
|
||||
// TODO - This is not currently possible (and we need to be aware of this).
|
||||
// Currently our SerIterator is limited to using a ReadTransaction only.
|
||||
//
|
||||
// Check we can see the new entry via an iterator using the uncommitted batch.
|
||||
// let mut iter: SerIterator<Vec<u8>> = batch.iter(&[0])?;
|
||||
// assert_eq!(iter.next(), Some((key.to_vec(), value.to_vec())));
|
||||
// assert_eq!(iter.next(), None);
|
||||
{
|
||||
let mut iter = batch.iter(Some(prefix), |_, v| Ok(v.to_vec()))?;
|
||||
assert_eq!(iter.next(), Some(Ok(value.to_vec())));
|
||||
assert_eq!(iter.next(), None);
|
||||
}
|
||||
|
||||
// Check we can not yet see the new entry via an iterator outside the uncommitted batch.
|
||||
let mut iter = store.iter(Some(prefix), |_, v| Ok(v.to_vec()))?;
|
||||
|
||||
Reference in New Issue
Block a user