sui_types/
full_checkpoint_content.rsuse std::collections::{BTreeMap, HashSet};
use crate::base_types::{ObjectID, ObjectRef};
use crate::effects::{TransactionEffects, TransactionEffectsAPI, TransactionEvents};
use crate::messages_checkpoint::{CertifiedCheckpointSummary, CheckpointContents};
use crate::object::Object;
use crate::storage::BackingPackageStore;
use crate::transaction::Transaction;
use serde::{Deserialize, Serialize};
use tap::Pipe;
#[derive(Clone, Debug, Serialize, Deserialize)]
pub struct CheckpointData {
pub checkpoint_summary: CertifiedCheckpointSummary,
pub checkpoint_contents: CheckpointContents,
pub transactions: Vec<CheckpointTransaction>,
}
impl CheckpointData {
pub fn latest_live_output_objects(&self) -> Vec<&Object> {
let mut latest_live_objects = BTreeMap::new();
for tx in self.transactions.iter() {
for obj in tx.output_objects.iter() {
latest_live_objects.insert(obj.id(), obj);
}
for obj_ref in tx.removed_object_refs_post_version() {
latest_live_objects.remove(&(obj_ref.0));
}
}
latest_live_objects.into_values().collect()
}
pub fn eventually_removed_object_refs_post_version(&self) -> Vec<ObjectRef> {
let mut eventually_removed_object_refs = BTreeMap::new();
for tx in self.transactions.iter() {
for obj_ref in tx.removed_object_refs_post_version() {
eventually_removed_object_refs.insert(obj_ref.0, obj_ref);
}
for obj in tx.output_objects.iter() {
eventually_removed_object_refs.remove(&(obj.id()));
}
}
eventually_removed_object_refs.into_values().collect()
}
pub fn checkpoint_input_objects(&self) -> BTreeMap<ObjectID, &Object> {
let mut output_objects_seen = HashSet::new();
let mut checkpoint_input_objects = BTreeMap::new();
for tx in self.transactions.iter() {
for obj in tx.input_objects.iter() {
let id = obj.id();
if output_objects_seen.contains(&id) || checkpoint_input_objects.contains_key(&id) {
continue;
}
checkpoint_input_objects.insert(id, obj);
}
for obj in tx.output_objects.iter() {
output_objects_seen.insert(obj.id());
}
}
checkpoint_input_objects
}
pub fn all_objects(&self) -> Vec<&Object> {
self.transactions
.iter()
.flat_map(|tx| &tx.input_objects)
.chain(self.transactions.iter().flat_map(|tx| &tx.output_objects))
.collect()
}
}
#[derive(Clone, Debug, Serialize, Deserialize)]
pub struct CheckpointTransaction {
pub transaction: Transaction,
pub effects: TransactionEffects,
pub events: Option<TransactionEvents>,
pub input_objects: Vec<Object>,
pub output_objects: Vec<Object>,
}
impl CheckpointTransaction {
pub fn removed_objects_pre_version(&self) -> impl Iterator<Item = &Object> {
self.effects
.all_removed_objects()
.into_iter() .map(|((id, _, _), _)| {
self.input_objects
.iter()
.find(|o| o.id() == id)
.expect("all removed objects should show up in input objects")
})
}
pub fn removed_object_refs_post_version(&self) -> impl Iterator<Item = ObjectRef> {
let deleted = self.effects.deleted().into_iter();
let wrapped = self.effects.wrapped().into_iter();
let unwrapped_then_deleted = self.effects.unwrapped_then_deleted().into_iter();
deleted.chain(wrapped).chain(unwrapped_then_deleted)
}
pub fn changed_objects(&self) -> impl Iterator<Item = (&Object, Option<&Object>)> {
self.effects
.all_changed_objects()
.into_iter()
.map(|((id, _, _), ..)| {
let object = self
.output_objects
.iter()
.find(|o| o.id() == id)
.expect("changed objects should show up in output objects");
let old_object = self.input_objects.iter().find(|o| o.id() == id);
(object, old_object)
})
}
pub fn created_objects(&self) -> impl Iterator<Item = &Object> {
self.effects
.created()
.into_iter()
.map(|((id, version, _), _)| {
self.output_objects
.iter()
.find(|o| o.id() == id && o.version() == version)
.expect("created objects should show up in output objects")
})
}
}
impl BackingPackageStore for CheckpointData {
fn get_package_object(
&self,
package_id: &crate::base_types::ObjectID,
) -> crate::error::SuiResult<Option<crate::storage::PackageObject>> {
self.transactions
.iter()
.flat_map(|transaction| transaction.output_objects.iter())
.find(|object| object.is_package() && &object.id() == package_id)
.cloned()
.map(crate::storage::PackageObject::new)
.pipe(Ok)
}
}