1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63
// Copyright (c) Mysten Labs, Inc.
// SPDX-License-Identifier: Apache-2.0
use once_cell::sync::OnceCell;
use serde::{Deserialize, Serialize};
use std::collections::HashSet;
use sui_types::base_types::TransactionDigest;
#[derive(Clone, Debug, Default, Deserialize, Serialize)]
#[serde(rename_all = "kebab-case")]
pub struct CertificateDenyConfig {
/// A list of certificate digests that are known to be either deterministically crashing
/// every validator, or causing every validator to hang forever, i.e. there is no way
/// for such transaction to execute successfully today.
/// Now with this config, a validator will decide that this transaction will always yield
/// ExecutionError and charge gas accordingly.
/// This config is meant for a fast temporary fix for a known issue, and should be removed
/// once the issue is fixed. However, since a certificate once executed will be included
/// in checkpoints, all future executions of this transaction through replay must also lead
/// to the same result (i.e. ExecutionError). So when we remove this config, we need to make
/// sure it's added to the constant certificate deny list in the Rust code (TODO: code link).
#[serde(default, skip_serializing_if = "Vec::is_empty")]
certificate_deny_list: Vec<TransactionDigest>,
/// In-memory cache for faster lookup of the certificate deny list.
#[serde(skip)]
certificate_deny_set: OnceCell<HashSet<TransactionDigest>>,
}
impl CertificateDenyConfig {
pub fn new() -> Self {
Self::default()
}
pub fn certificate_deny_set(&self) -> &HashSet<TransactionDigest> {
self.certificate_deny_set.get_or_init(|| {
self.certificate_deny_list
.iter()
.cloned()
.collect::<HashSet<_>>()
})
}
}
#[derive(Default)]
pub struct CertificateDenyConfigBuilder {
config: CertificateDenyConfig,
}
impl CertificateDenyConfigBuilder {
pub fn new() -> Self {
Self::default()
}
pub fn build(self) -> CertificateDenyConfig {
self.config
}
pub fn add_certificate_deny(mut self, certificate: TransactionDigest) -> Self {
self.config.certificate_deny_list.push(certificate);
self
}
}