signature 3.0.0

Traits for cryptographic signature algorithms (e.g. ECDSA, Ed25519)
Documentation
//! Traits for generating digital signatures.

use crate::error::Error;

#[cfg(feature = "digest")]
use crate::digest::Update;

#[cfg(feature = "rand_core")]
use crate::rand_core::{CryptoRng, TryCryptoRng};

/// Sign the provided message bytestring using `Self` (e.g. a cryptographic key or connection to an
/// HSM), returning a digital signature.
pub trait Signer<S> {
    /// Sign the given message and return a digital signature.
    fn sign(&self, msg: &[u8]) -> S {
        self.try_sign(msg).expect("signature operation failed")
    }

    /// Attempt to sign the given message, returning a digital signature on success, or an error if
    /// something went wrong.
    ///
    /// The main intended use case for signing errors is when communicating with external signers,
    /// e.g. cloud KMS, HSMs, or other hardware tokens.
    ///
    /// # Errors
    /// Returns implementation-specific errors in the event signing failed (e.g. KMS or HSM
    /// communication error).
    fn try_sign(&self, msg: &[u8]) -> Result<S, Error>;
}

/// Equivalent of [`Signer`] but the message is provided in non-contiguous byte slices.
pub trait MultipartSigner<S> {
    /// Equivalent of [`Signer::sign()`] but the message is provided in non-contiguous byte slices.
    fn multipart_sign(&self, msg: &[&[u8]]) -> S {
        self.try_multipart_sign(msg)
            .expect("signature operation failed")
    }

    /// Equivalent of [`Signer::try_sign()`] but the message is provided in non-contiguous byte
    /// slices.
    ///
    /// # Errors
    /// Returns implementation-specific errors in the event signing failed (e.g. KMS or HSM
    /// communication error).
    fn try_multipart_sign(&self, msg: &[&[u8]]) -> Result<S, Error>;
}

/// Sign the provided message bytestring using `&mut Self` (e.g. an evolving cryptographic key such
/// as a stateful hash-based signature), returning a digital signature.
pub trait SignerMut<S> {
    /// Sign the given message, update the state, and return a digital signature.
    ///
    /// # Panics
    /// In the event of a signing error.
    fn sign(&mut self, msg: &[u8]) -> S {
        self.try_sign(msg).expect("signature operation failed")
    }

    /// Attempt to sign the given message, updating the state, and returning a digital signature on
    /// success, or an error if something went wrong.
    ///
    /// # Errors
    /// Signing can fail, e.g. if the number of time periods allowed by the current key is exceeded.
    fn try_sign(&mut self, msg: &[u8]) -> Result<S, Error>;
}

/// Sign the given prehashed message `Digest` using `Self`.
///
/// ## Notes
///
/// This trait is primarily intended for signature algorithms based on the [Fiat-Shamir heuristic],
/// a method for converting an interactive challenge/response-based proof-of-knowledge protocol into
/// an offline digital signature through the use of a random oracle, i.e. a digest function.
///
/// The security of such protocols critically rests upon the inability of an attacker to solve for
/// the output of the random oracle, as generally otherwise such signature algorithms are a system
/// of linear equations and therefore doing so would allow the attacker to trivially forge
/// signatures.
///
/// To prevent misuse which would potentially allow this to be possible, this API accepts a `Digest`
/// instance, rather than a raw digest value.
///
/// [Fiat-Shamir heuristic]: https://en.wikipedia.org/wiki/Fiat%E2%80%93Shamir_heuristic
#[cfg(feature = "digest")]
pub trait DigestSigner<D: Update, S> {
    /// Sign a message by updating the received `Digest` with it, returning a signature.
    ///
    /// The given function can be invoked multiple times. It is expected that in each invocation the
    /// `Digest` is updated with the entire equal message.
    ///
    /// # Panics
    /// In the event of a signing error.
    fn sign_digest<F: Fn(&mut D)>(&self, f: F) -> S {
        self.try_sign_digest(|digest| {
            f(digest);
            Ok(())
        })
        .expect("signature operation failed")
    }

    /// Attempt to sign a message by updating the received `Digest` with it, returning a digital
    /// signature on success, or an error if something went wrong.
    ///
    /// The given function can be invoked multiple times. It is expected that in each invocation the
    /// `Digest` is updated with the entire equal message.
    ///
    /// # Errors
    /// Returns implementation-specific errors in the event signing failed (e.g. KMS or HSM
    /// communication error).
    fn try_sign_digest<F: Fn(&mut D) -> Result<(), Error>>(&self, f: F) -> Result<S, Error>;
}

/// Sign the given message using the provided external randomness source.
#[cfg(feature = "rand_core")]
pub trait RandomizedSigner<S> {
    /// Sign the given message and return a digital signature
    fn sign_with_rng<R: CryptoRng + ?Sized>(&self, rng: &mut R, msg: &[u8]) -> S {
        self.try_sign_with_rng(rng, msg)
            .expect("signature operation failed")
    }

    /// Attempt to sign the given message, returning a digital signature on success, or an error if
    /// something went wrong.
    ///
    /// The main intended use case for signing errors is when communicating with external signers,
    /// e.g. cloud KMS, HSMs, or other hardware tokens.
    ///
    /// # Errors
    /// Returns implementation-specific errors in the event signing failed (e.g. KMS or HSM
    /// communication error), or if the provided `rng` experiences an internal failure.
    fn try_sign_with_rng<R: TryCryptoRng + ?Sized>(
        &self,
        rng: &mut R,
        msg: &[u8],
    ) -> Result<S, Error>;
}

/// Equivalent of [`RandomizedSigner`] but the message is provided in non-contiguous byte slices.
#[cfg(feature = "rand_core")]
pub trait RandomizedMultipartSigner<S> {
    /// Equivalent of [`RandomizedSigner::sign_with_rng()`] but the message is provided in
    /// non-contiguous byte slices.
    fn multipart_sign_with_rng<R: CryptoRng + ?Sized>(&self, rng: &mut R, msg: &[&[u8]]) -> S {
        self.try_multipart_sign_with_rng(rng, msg)
            .expect("signature operation failed")
    }

    /// Equivalent of [`RandomizedSigner::try_sign_with_rng()`] but the message is provided in
    /// non-contiguous byte slices.
    ///
    /// # Errors
    /// Returns implementation-specific errors in the event signing failed (e.g. KMS or HSM
    /// communication error), or if the provided `rng` experiences an internal failure.
    fn try_multipart_sign_with_rng<R: TryCryptoRng + ?Sized>(
        &self,
        rng: &mut R,
        msg: &[&[u8]],
    ) -> Result<S, Error>;
}

/// Combination of [`DigestSigner`] and [`RandomizedSigner`] with support for computing a signature
/// over a digest which requires entropy from an RNG.
#[cfg(all(feature = "digest", feature = "rand_core"))]
pub trait RandomizedDigestSigner<D: Update, S> {
    /// Sign a message by updating the received `Digest` with it, returning a signature.
    ///
    /// The given function can be invoked multiple times. It is expected that in each invocation the
    /// `Digest` is updated with the entire equal message.
    ///
    /// # Panics
    /// In the event of a signing error.
    fn sign_digest_with_rng<R: CryptoRng + ?Sized, F: Fn(&mut D)>(&self, rng: &mut R, f: F) -> S {
        self.try_sign_digest_with_rng(rng, |digest| {
            f(digest);
            Ok(())
        })
        .expect("signature operation failed")
    }

    /// Attempt to sign a message by updating the received `Digest` with it, returning a digital
    /// signature on success, or an error if something went wrong.
    ///
    /// The given function can be invoked multiple times. It is expected that in each invocation the
    /// `Digest` is updated with the entire equal message.
    ///
    /// # Errors
    /// Returns implementation-specific errors in the event signing failed (e.g. KMS or HSM
    /// communication error), or if the provided `rng` experiences an internal failure.
    fn try_sign_digest_with_rng<R: TryCryptoRng + ?Sized, F: Fn(&mut D) -> Result<(), Error>>(
        &self,
        rng: &mut R,
        f: F,
    ) -> Result<S, Error>;
}

/// Sign the provided message bytestring using `&mut Self` (e.g. an evolving cryptographic key such
/// as a stateful hash-based signature), and a per-signature randomizer, returning a digital
/// signature.
#[cfg(feature = "rand_core")]
pub trait RandomizedSignerMut<S> {
    /// Sign the given message, update the state, and return a digital signature.
    ///
    /// # Panics
    /// In the event of a signing error.
    fn sign_with_rng<R: CryptoRng + ?Sized>(&mut self, rng: &mut R, msg: &[u8]) -> S {
        self.try_sign_with_rng(rng, msg)
            .expect("signature operation failed")
    }

    /// Attempt to sign the given message, updating the state, and returning a digital signature on
    /// success, or an error if something went wrong.
    ///
    /// # Errors
    /// Signing can fail, e.g. if the number of time periods allowed by the current key is exceeded,
    /// or if the provided `rng` experiences an internal failure.
    fn try_sign_with_rng<R: TryCryptoRng + ?Sized>(
        &mut self,
        rng: &mut R,
        msg: &[u8],
    ) -> Result<S, Error>;
}

/// Equivalent of [`RandomizedSignerMut`] but the message is provided in non-contiguous byte slices.
#[cfg(feature = "rand_core")]
pub trait RandomizedMultipartSignerMut<S> {
    /// Equivalent of [`RandomizedSignerMut::sign_with_rng()`] but the message is provided in
    /// non-contiguous byte slices.
    ///
    /// # Panics
    /// In the event of a signing error.
    fn multipart_sign_with_rng<R: CryptoRng + ?Sized>(&mut self, rng: &mut R, msg: &[&[u8]]) -> S {
        self.try_multipart_sign_with_rng(rng, msg)
            .expect("signature operation failed")
    }

    /// Equivalent of [`RandomizedSignerMut::try_sign_with_rng()`] but the message is provided in
    /// non-contiguous byte slices.
    ///
    /// # Errors
    /// Signing can fail, e.g. if the number of time periods allowed by the current key is exceeded,
    /// or if the provided `rng` experiences an internal failure.
    fn try_multipart_sign_with_rng<R: TryCryptoRng + ?Sized>(
        &mut self,
        rng: &mut R,
        msg: &[&[u8]],
    ) -> Result<S, Error>;
}

/// Blanket impl of [`RandomizedSignerMut`] for all [`RandomizedSigner`] types.
#[cfg(feature = "rand_core")]
impl<S, T: RandomizedSigner<S>> RandomizedSignerMut<S> for T {
    fn try_sign_with_rng<R: TryCryptoRng + ?Sized>(
        &mut self,
        rng: &mut R,
        msg: &[u8],
    ) -> Result<S, Error> {
        T::try_sign_with_rng(self, rng, msg)
    }
}

/// Asynchronously sign the provided message bytestring using `Self` (e.g. client for a Cloud KMS or
/// HSM), returning a digital signature.
///
/// This trait is an async equivalent of the [`Signer`] trait.
pub trait AsyncSigner<S> {
    /// Attempt to sign the given message, returning a digital signature on success, or an error if
    /// something went wrong.
    ///
    /// The main intended use case for signing errors is when communicating with external signers,
    /// e.g. cloud KMS, HSMs, or other hardware tokens.
    ///
    /// # Errors
    /// Returns implementation-specific errors in the event signing failed (e.g. KMS or HSM
    /// communication error).
    async fn sign_async(&self, msg: &[u8]) -> Result<S, Error>;
}

impl<S, T> AsyncSigner<S> for T
where
    T: Signer<S>,
{
    async fn sign_async(&self, msg: &[u8]) -> Result<S, Error> {
        self.try_sign(msg)
    }
}

/// Asynchronously sign the given prehashed message `Digest` using `Self`.
///
/// This trait is an async equivalent of the [`DigestSigner`] trait.
#[cfg(feature = "digest")]
pub trait AsyncDigestSigner<D, S>
where
    D: Update,
{
    /// Attempt to sign a message by updating the received `Digest` with it, returning a digital
    /// signature on success, or an error if something went wrong.
    ///
    /// The given function can be invoked multiple times. It is expected that in each invocation the
    /// `Digest` is updated with the entire equal message.
    ///
    /// # Errors
    /// Returns implementation-specific errors in the event signing failed (e.g. KMS or HSM
    /// communication error).
    async fn sign_digest_async<F: AsyncFn(&mut D) -> Result<(), Error>>(
        &self,
        f: F,
    ) -> Result<S, Error>;
}

/// Sign the given message using the provided external randomness source.
#[cfg(feature = "rand_core")]
pub trait AsyncRandomizedSigner<S> {
    /// Sign the given message and return a digital signature.
    ///
    /// # Panics
    /// In the event of a signing error.
    async fn sign_with_rng_async<R: CryptoRng + ?Sized>(&self, rng: &mut R, msg: &[u8]) -> S {
        self.try_sign_with_rng_async(rng, msg)
            .await
            .expect("signature operation failed")
    }

    /// Attempt to sign the given message, returning a digital signature on success, or an error if
    /// something went wrong.
    ///
    /// The main intended use case for signing errors is when communicating with external signers,
    /// e.g. cloud KMS, HSMs, or other hardware tokens.
    ///
    /// # Errors
    /// Returns implementation-specific errors in the event signing failed (e.g. KMS or HSM
    /// communication error), or if the provided `rng` experiences an internal failure.
    async fn try_sign_with_rng_async<R: TryCryptoRng + ?Sized>(
        &self,
        rng: &mut R,
        msg: &[u8],
    ) -> Result<S, Error>;
}

#[cfg(feature = "rand_core")]
impl<S, T> AsyncRandomizedSigner<S> for T
where
    T: RandomizedSigner<S>,
{
    async fn try_sign_with_rng_async<R: TryCryptoRng + ?Sized>(
        &self,
        rng: &mut R,
        msg: &[u8],
    ) -> Result<S, Error> {
        self.try_sign_with_rng(rng, msg)
    }
}