// This file is part of Substrate.
// Copyright (C) 2018-2022 Parity Technologies (UK) Ltd.
// SPDX-License-Identifier: GPL-3.0-or-later WITH Classpath-exception-2.0
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
// You should have received a copy of the GNU General Public License
// along with this program. If not, see .
//! Chain api required for the transaction pool.
use codec::Encode;
use futures::{
channel::{mpsc, oneshot},
future::{ready, Future, FutureExt, Ready},
lock::Mutex,
SinkExt, StreamExt,
};
use std::{marker::PhantomData, pin::Pin, sync::Arc};
use prometheus_endpoint::Registry as PrometheusRegistry;
use sc_client_api::{blockchain::HeaderBackend, BlockBackend};
use sp_api::{ApiExt, ProvideRuntimeApi};
use sp_core::traits::SpawnEssentialNamed;
use sp_runtime::{
generic::BlockId,
traits::{self, Block as BlockT, BlockIdTo},
transaction_validity::{TransactionSource, TransactionValidity},
};
use sp_transaction_pool::runtime_api::TaggedTransactionQueue;
use crate::{
error::{self, Error},
graph,
metrics::{ApiMetrics, ApiMetricsExt},
};
/// The transaction pool logic for full client.
pub struct FullChainApi {
client: Arc,
_marker: PhantomData,
metrics: Option>,
validation_pool: Arc + Send>>>>>,
}
/// Spawn a validation task that will be used by the transaction pool to validate transactions.
fn spawn_validation_pool_task(
name: &'static str,
receiver: Arc + Send>>>>>,
spawner: &impl SpawnEssentialNamed,
) {
spawner.spawn_essential_blocking(
name,
Some("transaction-pool"),
async move {
loop {
let task = receiver.lock().await.next().await;
match task {
None => return,
Some(task) => task.await,
}
}
}
.boxed(),
);
}
impl FullChainApi {
/// Create new transaction pool logic.
pub fn new(
client: Arc,
prometheus: Option<&PrometheusRegistry>,
spawner: &impl SpawnEssentialNamed,
) -> Self {
let metrics = prometheus.map(ApiMetrics::register).and_then(|r| match r {
Err(err) => {
log::warn!(
target: "txpool",
"Failed to register transaction pool api prometheus metrics: {:?}",
err,
);
None
},
Ok(api) => Some(Arc::new(api)),
});
let (sender, receiver) = mpsc::channel(0);
let receiver = Arc::new(Mutex::new(receiver));
spawn_validation_pool_task("transaction-pool-task-0", receiver.clone(), spawner);
spawn_validation_pool_task("transaction-pool-task-1", receiver, spawner);
FullChainApi {
client,
validation_pool: Arc::new(Mutex::new(sender)),
_marker: Default::default(),
metrics,
}
}
}
impl graph::ChainApi for FullChainApi
where
Block: BlockT,
Client:
ProvideRuntimeApi + BlockBackend + BlockIdTo + HeaderBackend,
Client: Send + Sync + 'static,
Client::Api: TaggedTransactionQueue,
{
type Block = Block;
type Error = error::Error;
type ValidationFuture =
Pin> + Send>>;
type BodyFuture = Ready::Extrinsic>>>>;
fn block_body(&self, id: &BlockId) -> Self::BodyFuture {
ready(self.client.block_body(id).map_err(error::Error::from))
}
fn validate_transaction(
&self,
at: &BlockId,
source: TransactionSource,
uxt: graph::ExtrinsicFor,
) -> Self::ValidationFuture {
let (tx, rx) = oneshot::channel();
let client = self.client.clone();
let at = *at;
let validation_pool = self.validation_pool.clone();
let metrics = self.metrics.clone();
async move {
metrics.report(|m| m.validations_scheduled.inc());
validation_pool
.lock()
.await
.send(
async move {
let res = validate_transaction_blocking(&*client, &at, source, uxt);
let _ = tx.send(res);
metrics.report(|m| m.validations_finished.inc());
}
.boxed(),
)
.await
.map_err(|e| Error::RuntimeApi(format!("Validation pool down: {:?}", e)))?;
match rx.await {
Ok(r) => r,
Err(_) => Err(Error::RuntimeApi("Validation was canceled".into())),
}
}
.boxed()
}
fn block_id_to_number(
&self,
at: &BlockId,
) -> error::Result