use crate::{
prover::{BaseProveRequest, SendFutureResult},
CpuProver, LightProver, MockProver, Prover, SP1ProofWithPublicValues, SP1VerificationError,
StatusCode,
};
#[cfg(feature = "cuda")]
use crate::{cuda::builder::CudaProverBuilder, CudaProver};
use sp1_core_executor::SP1CoreOpts;
#[cfg(feature = "network")]
use crate::NetworkProver;
pub mod pk;
pub mod prove;
pub use pk::EnvProvingKey;
use prove::EnvProveRequest;
use sp1_core_machine::io::SP1Stdin;
use sp1_core_machine::riscv::RiscvAir;
use sp1_hypercube::Machine;
use sp1_primitives::{Elf, SP1Field};
use sp1_prover::{worker::SP1NodeCore, SP1VerifyingKey};
#[derive(Clone)]
#[allow(clippy::large_enum_variant)]
pub enum EnvProver {
Mock(MockProver),
Light(LightProver),
Cpu(CpuProver),
#[cfg(feature = "cuda")]
Cuda(CudaProver),
#[cfg(feature = "network")]
Network(NetworkProver),
}
impl EnvProver {
pub async fn new() -> Self {
Self::new_with_machine(RiscvAir::machine()).await
}
pub async fn new_with_machine(machine: Machine<SP1Field, RiscvAir<SP1Field>>) -> Self {
Self::from_env_with_opts_and_machine(None, machine).await
}
pub async fn with_opts(self, opts: SP1CoreOpts) -> Self {
Self::from_env_with_opts(Some(opts)).await
}
pub async fn from_env_with_opts(core_opts: Option<SP1CoreOpts>) -> Self {
Self::from_env_with_opts_and_machine(core_opts, RiscvAir::machine()).await
}
pub async fn from_env_with_opts_and_machine(
core_opts: Option<SP1CoreOpts>,
machine: Machine<SP1Field, RiscvAir<SP1Field>>,
) -> Self {
let prover = match std::env::var("SP1_PROVER") {
Ok(prover) => prover,
Err(_) => "cpu".to_string(),
};
match prover.as_str() {
"cpu" => Self::Cpu(CpuProver::new_with_opts_and_machine(core_opts, machine).await),
#[cfg(feature = "cuda")]
"cuda" => Self::Cuda(CudaProverBuilder::new_with_machine(machine).build().await),
#[cfg(not(feature = "cuda"))]
"cuda" => panic!("The CUDA prover requires the `cuda` feature to be enabled"),
"mock" => Self::Mock(MockProver::new_with_machine(machine).await),
"light" => Self::Light(LightProver::new_with_machine(machine).await),
#[cfg(feature = "network")]
"network" | "hosted" => {
let private_key =
std::env::var("NETWORK_PRIVATE_KEY").ok().filter(|k| !k.is_empty()).expect(
"NETWORK_PRIVATE_KEY environment variable is not set. \
Please set it to your private key or use the .private_key() method.",
);
let mut network_builder =
crate::network::builder::NetworkProverBuilder::new_with_machine(machine)
.private_key(&private_key);
if prover == "hosted" {
network_builder = network_builder.hosted();
}
Self::Network(network_builder.build().await)
}
_ => unreachable!(),
}
}
}
impl Prover for EnvProver {
type Error = anyhow::Error;
type ProvingKey = EnvProvingKey;
type ProveRequest<'a> = prove::EnvProveRequest<'a>;
fn inner(&self) -> &SP1NodeCore {
match self {
Self::Cpu(prover) => prover.inner(),
#[cfg(feature = "cuda")]
Self::Cuda(prover) => prover.inner(),
Self::Mock(prover) => prover.inner(),
Self::Light(prover) => prover.inner(),
#[cfg(feature = "network")]
Self::Network(prover) => prover.inner(),
}
}
fn setup(&self, elf: Elf) -> impl SendFutureResult<Self::ProvingKey, Self::Error> {
async move {
match self {
Self::Cpu(prover) => {
let pk = prover.setup(elf).await?;
Ok(EnvProvingKey::cpu(pk))
}
#[cfg(feature = "cuda")]
Self::Cuda(prover) => {
let pk = prover.setup(elf).await?;
Ok(EnvProvingKey::cuda(pk))
}
Self::Mock(prover) => {
let pk = prover.setup(elf).await?;
Ok(EnvProvingKey::mock(pk))
}
Self::Light(prover) => {
let pk = prover.setup(elf).await?;
Ok(EnvProvingKey::light(pk))
}
#[cfg(feature = "network")]
Self::Network(prover) => {
let pk = prover.setup(elf).await?;
Ok(EnvProvingKey::network(pk))
}
}
}
}
fn prove<'a>(&'a self, pk: &'a Self::ProvingKey, stdin: SP1Stdin) -> Self::ProveRequest<'a> {
EnvProveRequest { base: BaseProveRequest::new(self, pk, stdin) }
}
fn verify(
&self,
proof: &SP1ProofWithPublicValues,
vkey: &SP1VerifyingKey,
status_code: Option<StatusCode>,
) -> Result<(), SP1VerificationError> {
match self {
Self::Cpu(prover) => prover.verify(proof, vkey, status_code),
#[cfg(feature = "cuda")]
Self::Cuda(prover) => prover.verify(proof, vkey, status_code),
Self::Mock(prover) => prover.verify(proof, vkey, status_code),
Self::Light(prover) => prover.verify(proof, vkey, status_code),
#[cfg(feature = "network")]
Self::Network(prover) => prover.verify(proof, vkey, status_code),
}
}
}
#[cfg(test)]
mod tests {
use crate::{prover::ProveRequest, utils::setup_logger, MockProver, Prover, SP1Stdin};
use super::EnvProver;
#[tokio::test]
async fn test_envprover_mock_verifies_plonk_and_groth16() {
setup_logger();
let mock = MockProver::new().await;
let pk =
mock.setup(test_artifacts::FIBONACCI_ELF).await.expect("failed to setup proving key");
let mut stdin = SP1Stdin::new();
stdin.write(&10usize);
let plonk_proof =
mock.prove(&pk, stdin).plonk().await.expect("failed to create mock Plonk proof");
let mut stdin = SP1Stdin::new();
stdin.write(&10usize);
let groth16_proof =
mock.prove(&pk, stdin).groth16().await.expect("failed to create mock Groth16 proof");
let env = EnvProver::Mock(mock);
env.verify(&plonk_proof, &pk.vk, None)
.expect("EnvProver::Mock must verify a mock Plonk proof");
env.verify(&groth16_proof, &pk.vk, None)
.expect("EnvProver::Mock must verify a mock Groth16 proof");
}
}