1use crate::{epoch::Epoch, state_key};
2use anyhow::{anyhow, Context, Result};
3use async_trait::async_trait;
4use cnidarium::{StateRead, StateWrite};
5use penumbra_sdk_proto::{StateReadProto, StateWriteProto};
6use std::str::FromStr;
78#[async_trait]
9/// Provides read access to epoch indices, block heights, timestamps, and other related data.
10pub trait EpochRead: StateRead {
11/// Get the current block height.
12 ///
13 /// # Errors
14 /// Returns an error if the block height is missing.
15async fn get_block_height(&self) -> Result<u64> {
16self.get_proto(state_key::block_manager::block_height())
17 .await?
18.ok_or_else(|| anyhow!("Missing block_height"))
19 }
2021/// Gets the current block timestamp from the JMT
22 ///
23 /// # Errors
24 /// Returns an error if the block timestamp is missing.
25 ///
26 /// # Panic
27 /// Panics if the block timestamp is not a valid RFC3339 time string.
28async fn get_current_block_timestamp(&self) -> Result<tendermint::Time> {
29let timestamp_string: String = self
30.get_proto(state_key::block_manager::current_block_timestamp())
31 .await?
32.ok_or_else(|| anyhow!("Missing current_block_timestamp"))?;
3334Ok(tendermint::Time::from_str(×tamp_string)
35 .context("current_block_timestamp was an invalid RFC3339 time string")?)
36 }
3738/// Gets a historic block timestamp from nonverifiable storage.
39 ///
40 /// # Errors
41 /// Returns an error if the block timestamp is missing.
42 ///
43 /// # Panic
44 /// Panics if the block timestamp is not a valid RFC3339 time string.
45async fn get_block_timestamp(&self, height: u64) -> Result<tendermint::Time> {
46let timestamp_string: String = self
47.nonverifiable_get_proto(&state_key::block_manager::block_timestamp(height).as_bytes())
48 .await?
49.ok_or_else(|| anyhow!("Missing block_timestamp for height {}", height))?;
5051Ok(
52 tendermint::Time::from_str(×tamp_string).context(format!(
53"block_timestamp for height {} was an invalid RFC3339 time string",
54 height
55 ))?,
56 )
57 }
5859/// Get the current application epoch.
60 ///
61 /// # Errors
62 /// Returns an error if the epoch is missing.
63async fn get_current_epoch(&self) -> Result<Epoch> {
64// Get the height
65let height = self.get_block_height().await?;
6667self.get(&state_key::epoch_manager::epoch_by_height(height))
68 .await?
69.ok_or_else(|| anyhow!("missing epoch for current height: {height}"))
70 }
7172/// Get the epoch corresponding to the supplied height.
73 ///
74 /// # Errors
75 /// Returns an error if the epoch is missing.
76async fn get_epoch_by_height(&self, height: u64) -> Result<Epoch> {
77self.get(&state_key::epoch_manager::epoch_by_height(height))
78 .await?
79.ok_or_else(|| anyhow!("missing epoch for height"))
80 }
8182/// Returns true if we are triggering an early epoch end.
83async fn is_epoch_ending_early(&self) -> bool {
84self.object_get(state_key::epoch_manager::end_epoch_early())
85 .unwrap_or(false)
86 }
87}
8889impl<T: StateRead + ?Sized> EpochRead for T {}
9091/// Provides write access to the chain's epoch manager.
92/// The epoch manager is responsible for tracking block and epoch heights
93/// as well as related data like reported timestamps and epoch duration.
94#[async_trait]
95pub trait EpochManager: StateWrite {
96/// Writes the current block's timestamp as an RFC3339 string to verifiable storage.
97 ///
98 /// Also writes the current block's timestamp to the appropriate key in nonverifiable storage.
99fn put_block_timestamp(&mut self, height: u64, timestamp: tendermint::Time) {
100self.put_proto(
101 state_key::block_manager::current_block_timestamp().into(),
102 timestamp.to_rfc3339(),
103 );
104105self.nonverifiable_put_proto(
106 state_key::block_manager::block_timestamp(height).into(),
107 timestamp.to_rfc3339(),
108 );
109 }
110111/// Write a value in the end epoch flag in object-storage.
112 /// This is used to trigger an early epoch end at the end of the block.
113fn set_end_epoch_flag(&mut self) {
114self.object_put(state_key::epoch_manager::end_epoch_early(), true)
115 }
116117/// Writes the block height to verifiable storage.
118fn put_block_height(&mut self, height: u64) {
119self.put_proto(state_key::block_manager::block_height().to_string(), height)
120 }
121122/// Index the current epoch by height.
123fn put_epoch_by_height(&mut self, height: u64, epoch: Epoch) {
124self.put(state_key::epoch_manager::epoch_by_height(height), epoch)
125 }
126}
127128impl<T: StateWrite + ?Sized> EpochManager for T {}