,
CIDP: CreateInherentDataProviders,
CIDP::InherentDataProviders: InherentDataProviderExt,
{
/// Create a new instance of AURA consensus.
pub fn new(
para_client: Arc,
block_import: BI,
sync_oracle: SO,
proposer_factory: PF,
force_authoring: bool,
backoff_authoring_blocks: Option,
keystore: SyncCryptoStorePtr,
create_inherent_data_providers: CIDP,
polkadot_client: Arc,
polkadot_backend: Arc,
slot_duration: SlotDuration,
telemetry: Option,
block_proposal_slot_portion: SlotProportion,
max_block_proposal_slot_portion: Option,
) -> Self
where
Client: ProvideRuntimeApi
+ BlockOf
+ ProvideCache
+ AuxStore
+ HeaderBackend
+ Send
+ Sync
+ 'static,
Client::Api: AuraApi,
BI: BlockImport> + Send + Sync + 'static,
SO: SyncOracle + Send + Sync + Clone + 'static,
BS: BackoffAuthoringBlocksStrategy> + Send + Sync + 'static,
PF: Environment + Send + Sync + 'static,
PF::Proposer: Proposer<
B,
Error = Error,
Transaction = sp_api::TransactionFor,
ProofRecording = EnableProofRecording,
Proof = ::Proof,
>,
Error: std::error::Error + Send + From + 'static,
P: Pair + Send + Sync,
P::Public: AppPublic + Hash + Member + Encode + Decode,
P::Signature: TryFrom> + Hash + Member + Encode + Decode,
{
let worker = sc_consensus_aura::build_aura_worker::(
BuildAuraWorkerParams {
client: para_client,
block_import: ParachainBlockImport::new(block_import),
justification_sync_link: (),
proposer_factory,
sync_oracle,
force_authoring,
backoff_authoring_blocks,
keystore,
telemetry,
block_proposal_slot_portion,
max_block_proposal_slot_portion,
},
);
Self {
create_inherent_data_providers: Arc::new(create_inherent_data_providers),
relay_chain_backend: polkadot_backend,
relay_chain_client: polkadot_client,
aura_worker: Arc::new(Mutex::new(worker)),
slot_duration,
}
}
/// Create the inherent data.
///
/// Returns the created inherent data and the inherent data providers used.
async fn inherent_data(
&self,
parent: B::Hash,
validation_data: &PersistedValidationData,
relay_parent: PHash,
) -> Option<(InherentData, CIDP::InherentDataProviders)> {
let inherent_data_providers = self
.create_inherent_data_providers
.create_inherent_data_providers(parent, (relay_parent, validation_data.clone()))
.await
.map_err(|e| {
tracing::error!(
target: LOG_TARGET,
error = ?e,
"Failed to create inherent data providers.",
)
})
.ok()?;
inherent_data_providers
.create_inherent_data()
.map_err(|e| {
tracing::error!(
target: LOG_TARGET,
error = ?e,
"Failed to create inherent data.",
)
})
.ok()
.map(|d| (d, inherent_data_providers))
}
}
#[async_trait::async_trait]
impl ParachainConsensus for AuraConsensus
where
B: BlockT,
RClient: ProvideRuntimeApi + Send + Sync,
RClient::Api: ParachainHost,
RBackend: Backend,
CIDP: CreateInherentDataProviders + Send + Sync,
CIDP::InherentDataProviders: InherentDataProviderExt + Send,
{
async fn produce_candidate(
&mut self,
parent: &B::Header,
relay_parent: PHash,
validation_data: &PersistedValidationData,
) -> Option> {
let (inherent_data, inherent_data_providers) = self
.inherent_data(parent.hash(), validation_data, relay_parent)
.await?;
let info = SlotInfo::new(
inherent_data_providers.slot(),
inherent_data_providers.timestamp(),
inherent_data,
self.slot_duration.slot_duration(),
parent.clone(),
// Set the block limit to 50% of the maximum PoV size.
//
// TODO: If we got benchmarking that includes the proof size,
// we should be able to use the maximum pov size.
Some((validation_data.max_pov_size / 2) as usize),
);
let res = self.aura_worker.lock().await.on_slot(info).await?;
Some(ParachainCandidate {
block: res.block,
proof: res.storage_proof,
})
}
}
/// Paramaters of [`build_aura_consensus`].
pub struct BuildAuraConsensusParams {
pub proposer_factory: PF,
pub create_inherent_data_providers: CIDP,
pub block_import: BI,
pub relay_chain_client: polkadot_client::Client,
pub relay_chain_backend: Arc,
pub para_client: Arc,
pub backoff_authoring_blocks: Option,
pub sync_oracle: SO,
pub keystore: SyncCryptoStorePtr,
pub force_authoring: bool,
pub slot_duration: SlotDuration,
pub telemetry: Option,
pub block_proposal_slot_portion: SlotProportion,
pub max_block_proposal_slot_portion: Option,
}
/// Build the [`AuraConsensus`].
///
/// Returns a boxed [`ParachainConsensus`].
pub fn build_aura_consensus(
BuildAuraConsensusParams {
proposer_factory,
create_inherent_data_providers,
block_import,
relay_chain_client,
relay_chain_backend,
para_client,
backoff_authoring_blocks,
sync_oracle,
keystore,
force_authoring,
slot_duration,
telemetry,
block_proposal_slot_portion,
max_block_proposal_slot_portion,
}: BuildAuraConsensusParams,
) -> Box>
where
Block: BlockT,
// Rust bug: https://github.com/rust-lang/rust/issues/24159
sc_client_api::StateBackendFor: sc_client_api::StateBackend>,
RBackend: Backend + 'static,
CIDP: CreateInherentDataProviders
+ Send
+ Sync
+ 'static,
CIDP::InherentDataProviders: InherentDataProviderExt + Send,
Client: ProvideRuntimeApi
+ BlockOf
+ ProvideCache
+ AuxStore
+ HeaderBackend
+ Send
+ Sync
+ 'static,
Client::Api: AuraApi,
BI: BlockImport>
+ Send
+ Sync
+ 'static,
SO: SyncOracle + Send + Sync + Clone + 'static,
BS: BackoffAuthoringBlocksStrategy> + Send + Sync + 'static,
PF: Environment + Send + Sync + 'static,
PF::Proposer: Proposer<
Block,
Error = Error,
Transaction = sp_api::TransactionFor,
ProofRecording = EnableProofRecording,
Proof = ::Proof,
>,
Error: std::error::Error + Send + From + 'static,
P: Pair + Send + Sync,
P::Public: AppPublic + Hash + Member + Encode + Decode,
P::Signature: TryFrom> + Hash + Member + Encode + Decode,
{
AuraConsensusBuilder::::new(
proposer_factory,
block_import,
create_inherent_data_providers,
relay_chain_client,
relay_chain_backend,
para_client,
backoff_authoring_blocks,
sync_oracle,
force_authoring,
keystore,
slot_duration,
telemetry,
block_proposal_slot_portion,
max_block_proposal_slot_portion,
)
.build()
}
/// Aura consensus builder.
///
/// Builds a [`AuraConsensus`] for a parachain. As this requires
/// a concrete relay chain client instance, the builder takes a [`polkadot_client::Client`]
/// that wraps this concrete instance. By using [`polkadot_client::ExecuteWithClient`]
/// the builder gets access to this concrete instance.
struct AuraConsensusBuilder
{
_phantom: PhantomData<(Block, Error, P)>,
proposer_factory: PF,
create_inherent_data_providers: CIDP,
block_import: BI,
relay_chain_backend: Arc,
relay_chain_client: polkadot_client::Client,
para_client: Arc,
backoff_authoring_blocks: Option,
sync_oracle: SO,
force_authoring: bool,
keystore: SyncCryptoStorePtr,
slot_duration: SlotDuration,
telemetry: Option,
block_proposal_slot_portion: SlotProportion,
max_block_proposal_slot_portion: Option,
}
impl
AuraConsensusBuilder
where
Block: BlockT,
// Rust bug: https://github.com/rust-lang/rust/issues/24159
sc_client_api::StateBackendFor: sc_client_api::StateBackend>,
RBackend: Backend + 'static,
CIDP: CreateInherentDataProviders
+ Send
+ Sync
+ 'static,
CIDP::InherentDataProviders: InherentDataProviderExt + Send,
Client: ProvideRuntimeApi
+ BlockOf
+ ProvideCache
+ AuxStore
+ HeaderBackend
+ Send
+ Sync
+ 'static,
Client::Api: AuraApi,
BI: BlockImport>
+ Send
+ Sync
+ 'static,
SO: SyncOracle + Send + Sync + Clone + 'static,
BS: BackoffAuthoringBlocksStrategy> + Send + Sync + 'static,
PF: Environment + Send + Sync + 'static,
PF::Proposer: Proposer<
Block,
Error = Error,
Transaction = sp_api::TransactionFor,
ProofRecording = EnableProofRecording,
Proof = ::Proof,
>,
Error: std::error::Error + Send + From + 'static,
P: Pair + Send + Sync,
P::Public: AppPublic + Hash + Member + Encode + Decode,
P::Signature: TryFrom> + Hash + Member + Encode + Decode,
{
/// Create a new instance of the builder.
fn new(
proposer_factory: PF,
block_import: BI,
create_inherent_data_providers: CIDP,
relay_chain_client: polkadot_client::Client,
relay_chain_backend: Arc,
para_client: Arc,
backoff_authoring_blocks: Option,
sync_oracle: SO,
force_authoring: bool,
keystore: SyncCryptoStorePtr,
slot_duration: SlotDuration,
telemetry: Option,
block_proposal_slot_portion: SlotProportion,
max_block_proposal_slot_portion: Option,
) -> Self {
Self {
_phantom: PhantomData,
proposer_factory,
block_import,
create_inherent_data_providers,
relay_chain_backend,
relay_chain_client,
para_client,
backoff_authoring_blocks,
sync_oracle,
force_authoring,
keystore,
slot_duration,
telemetry,
block_proposal_slot_portion,
max_block_proposal_slot_portion,
}
}
/// Build the relay chain consensus.
fn build(self) -> Box> {
self.relay_chain_client.clone().execute_with(self)
}
}
impl polkadot_client::ExecuteWithClient
for AuraConsensusBuilder
where
Block: BlockT,
// Rust bug: https://github.com/rust-lang/rust/issues/24159
sc_client_api::StateBackendFor: sc_client_api::StateBackend>,
RBackend: Backend + 'static,
CIDP: CreateInherentDataProviders
+ Send
+ Sync
+ 'static,
CIDP::InherentDataProviders: InherentDataProviderExt + Send,
Client: ProvideRuntimeApi
+ BlockOf
+ ProvideCache
+ AuxStore
+ HeaderBackend
+ Send
+ Sync
+ 'static,
Client::Api: AuraApi,
BI: BlockImport>
+ Send
+ Sync
+ 'static,
SO: SyncOracle + Send + Sync + Clone + 'static,
BS: BackoffAuthoringBlocksStrategy> + Send + Sync + 'static,
PF: Environment + Send + Sync + 'static,
PF::Proposer: Proposer<
Block,
Error = Error,
Transaction = sp_api::TransactionFor,
ProofRecording = EnableProofRecording,
Proof = ::Proof,
>,
Error: std::error::Error + Send + From + 'static,
P: Pair + Send + Sync,
P::Public: AppPublic + Hash + Member + Encode + Decode,
P::Signature: TryFrom> + Hash + Member + Encode + Decode,
{
type Output = Box>;
fn execute_with_client(self, client: Arc) -> Self::Output
where
>::StateBackend: sp_api::StateBackend>,
PBackend: Backend,
PBackend::State: sp_api::StateBackend,
Api: polkadot_client::RuntimeApiCollection,
PClient: polkadot_client::AbstractClient + 'static,
{
Box::new(AuraConsensus::new::(
self.para_client,
self.block_import,
self.sync_oracle,
self.proposer_factory,
self.force_authoring,
self.backoff_authoring_blocks,
self.keystore,
self.create_inherent_data_providers,
client.clone(),
self.relay_chain_backend,
self.slot_duration,
self.telemetry,
self.block_proposal_slot_portion,
self.max_block_proposal_slot_portion,
))
}
}