miden-remote-prover 0.15.0

Miden remote prover
use miden_node_proto::generated as grpc;
use miden_node_utils::tracing::OpenTelemetrySpanExt;

use crate::server::proof_kind::ProofKind;
use crate::server::service::ProverService;

#[tonic::async_trait]
impl grpc::server::remote_prover_api::Prove for ProverService {
    type Input = (ProofKind, grpc::remote_prover::ProofRequest);
    type Output = grpc::remote_prover::Proof;

    async fn handle(&self, (proof_kind, request): Self::Input) -> tonic::Result<Self::Output> {
        tracing::Span::current().set_attribute("request.kind", proof_kind);

        // Reject unsupported proof types early so they don't clog the queue.
        if !self.is_supported(proof_kind) {
            return Err(tonic::Status::invalid_argument("unsupported proof type"));
        }

        // This semaphore acts like a queue, but with a fixed capacity.
        //
        // We need to hold this until our request is processed to ensure that the queue capacity is
        // not exceeded.
        let _permit = self.acquire_permit()?;

        // This mutex is fair and uses FIFO ordering.
        let prover = self.acquire_prover().await;

        prover.prove(request).await
    }

    fn decode(request: grpc::remote_prover::ProofRequest) -> tonic::Result<Self::Input> {
        // Check that the proof type is supported. Protobuf enums return a default value if the enum
        // is set to an unknown value. This round trip checks that the value is valid.
        if request.proof_type() as i32 != request.proof_type {
            return Err(tonic::Status::invalid_argument("unknown proof_type value"));
        }

        Ok((ProofKind::from(request.proof_type()), request))
    }

    fn encode(output: Self::Output) -> tonic::Result<grpc::remote_prover::Proof> {
        Ok(output)
    }
}