use std::sync::atomic::Ordering;
use miden_node_proto::generated as grpc;
use miden_node_utils::ErrorReport;
use miden_node_utils::tracing::OpenTelemetrySpanExt;
use miden_protocol::transaction::{ProvenTransaction, TransactionInputs};
use miden_tx::utils::serde::Deserializable;
use tonic::Status;
use crate::db::insert_transaction;
use crate::server::ValidatorServer;
use crate::tx_validation::validate_transaction;
#[tonic::async_trait]
impl grpc::server::validator_api::SubmitProvenTransaction for ValidatorServer {
type Input = Input;
type Output = ();
async fn handle(&self, input: Self::Input) -> tonic::Result<Self::Output> {
tracing::Span::current().set_attribute("transaction.id", input.tx.id());
let tx_info = validate_transaction(input.tx, input.inputs).await.map_err(|err| {
Status::invalid_argument(err.as_report_context("Invalid transaction"))
})?;
let count = self
.db
.transact("insert_transaction", move |conn| insert_transaction(conn, &tx_info))
.await
.map_err(|err| {
Status::internal(err.as_report_context("Failed to insert transaction"))
})?;
self.validated_transactions_count.fetch_add(count as u64, Ordering::Relaxed);
Ok(())
}
fn decode(request: grpc::transaction::ProvenTransaction) -> tonic::Result<Self::Input> {
let tx = ProvenTransaction::read_from_bytes(&request.transaction).map_err(|err| {
Status::invalid_argument(err.as_report_context("Invalid proven transaction"))
})?;
let inputs = request
.transaction_inputs
.ok_or(Status::invalid_argument("Missing transaction inputs"))?;
let inputs = TransactionInputs::read_from_bytes(&inputs).map_err(|err| {
Status::invalid_argument(err.as_report_context("Invalid transaction inputs"))
})?;
Ok(Self::Input { tx, inputs })
}
fn encode(output: Self::Output) -> tonic::Result<()> {
Ok(output)
}
}
pub struct Input {
tx: ProvenTransaction,
inputs: TransactionInputs,
}