consensus_core/
context.rsuse std::{sync::Arc, time::SystemTime};
use consensus_config::{AuthorityIndex, Committee, Parameters};
#[cfg(test)]
use consensus_config::{NetworkKeyPair, ProtocolKeyPair};
use sui_protocol_config::ProtocolConfig;
#[cfg(test)]
use tempfile::TempDir;
use tokio::time::Instant;
#[cfg(test)]
use crate::metrics::test_metrics;
use crate::{block::BlockTimestampMs, metrics::Metrics};
#[derive(Clone)]
pub(crate) struct Context {
pub own_index: AuthorityIndex,
pub committee: Committee,
pub parameters: Parameters,
pub protocol_config: ProtocolConfig,
pub metrics: Arc<Metrics>,
pub clock: Arc<Clock>,
}
impl Context {
pub(crate) fn new(
own_index: AuthorityIndex,
committee: Committee,
parameters: Parameters,
protocol_config: ProtocolConfig,
metrics: Arc<Metrics>,
clock: Arc<Clock>,
) -> Self {
Self {
own_index,
committee,
parameters,
protocol_config,
metrics,
clock,
}
}
#[cfg(test)]
pub(crate) fn new_for_test(
committee_size: usize,
) -> (Self, Vec<(NetworkKeyPair, ProtocolKeyPair)>) {
let (committee, keypairs) =
consensus_config::local_committee_and_keys(0, vec![1; committee_size]);
let metrics = test_metrics();
let temp_dir = TempDir::new().unwrap();
let clock = Arc::new(Clock::default());
let context = Context::new(
AuthorityIndex::new_for_test(0),
committee,
Parameters {
db_path: temp_dir.into_path(),
..Default::default()
},
ProtocolConfig::get_for_max_version_UNSAFE(),
metrics,
clock,
);
(context, keypairs)
}
#[cfg(test)]
pub(crate) fn with_authority_index(mut self, authority: AuthorityIndex) -> Self {
self.own_index = authority;
self
}
#[cfg(test)]
pub(crate) fn with_committee(mut self, committee: Committee) -> Self {
self.committee = committee;
self
}
#[cfg(test)]
pub(crate) fn with_parameters(mut self, parameters: Parameters) -> Self {
self.parameters = parameters;
self
}
#[cfg(test)]
pub(crate) fn with_protocol_config(mut self, protocol_config: ProtocolConfig) -> Self {
self.protocol_config = protocol_config;
self
}
}
pub struct Clock {
initial_instant: Instant,
initial_system_time: SystemTime,
clock_drift: BlockTimestampMs,
}
impl Default for Clock {
fn default() -> Self {
Self {
initial_instant: Instant::now(),
initial_system_time: SystemTime::now(),
clock_drift: 0,
}
}
}
impl Clock {
pub fn new_for_test(clock_drift: BlockTimestampMs) -> Self {
Self {
initial_instant: Instant::now(),
initial_system_time: SystemTime::now(),
clock_drift,
}
}
pub(crate) fn timestamp_utc_ms(&self) -> BlockTimestampMs {
if cfg!(not(any(msim, test))) {
assert_eq!(
self.clock_drift, 0,
"Clock drift should not be set in non testing environments."
);
}
let now: Instant = Instant::now();
let monotonic_system_time = self
.initial_system_time
.checked_add(
now.checked_duration_since(self.initial_instant)
.unwrap_or_else(|| {
panic!(
"current instant ({:?}) < initial instant ({:?})",
now, self.initial_instant
)
}),
)
.expect("Computing system time should not overflow");
monotonic_system_time
.duration_since(SystemTime::UNIX_EPOCH)
.unwrap_or_else(|_| {
panic!(
"system time ({:?}) < UNIX_EPOCH ({:?})",
monotonic_system_time,
SystemTime::UNIX_EPOCH,
)
})
.as_millis() as BlockTimestampMs
+ self.clock_drift
}
}