tfhe 1.6.1

TFHE-rs is a fully homomorphic encryption (FHE) library that implements Zama's variant of TFHE.
Documentation
use crate::integer::{BooleanBlock, ServerKey as IntegerServerKey};
use crate::strings::ciphertext::{FheString, GenericPattern, GenericPatternRef};
use crate::strings::server_key::{FheStringIsEmpty, ServerKey};
use crate::ClearString;
use std::borrow::Borrow;

impl<T: Borrow<IntegerServerKey> + Sync> ServerKey<T> {
    fn eq_length_checks(&self, lhs: &FheString, rhs: &FheString) -> Option<BooleanBlock> {
        let sk = self.inner();

        // If lhs is empty, rhs must also be empty in order to be equal (the case where lhs is
        // empty with > 1 padding zeros is handled next)
        if lhs.is_empty() {
            return match self.is_empty(rhs) {
                FheStringIsEmpty::Padding(enc_val) => Some(enc_val),
                FheStringIsEmpty::NoPadding(val) => Some(sk.create_trivial_boolean_block(val)),
            };
        }

        // If rhs is empty, lhs must also be empty in order to be equal (only case remaining is if
        // lhs padding zeros > 1)
        if rhs.is_empty() {
            return match self.is_empty(lhs) {
                FheStringIsEmpty::Padding(enc_val) => Some(enc_val),
                FheStringIsEmpty::NoPadding(_) => Some(sk.create_trivial_boolean_block(false)),
            };
        }

        // Two strings without padding that have different lengths cannot be equal
        if (!lhs.is_padded() && !rhs.is_padded()) && (lhs.len() != rhs.len()) {
            return Some(sk.create_trivial_boolean_block(false));
        }

        // A string without padding cannot be equal to a string with padding that has the same or
        // lower length
        if (!lhs.is_padded() && rhs.is_padded()) && (rhs.len() <= lhs.len())
            || (!rhs.is_padded() && lhs.is_padded()) && (lhs.len() <= rhs.len())
        {
            return Some(sk.create_trivial_boolean_block(false));
        }

        None
    }

    /// Returns `true` if an encrypted string and a pattern (either encrypted or clear) are equal.
    ///
    /// Returns `false` if they are not equal.
    ///
    /// The pattern for comparison (`rhs`) can be specified as either `GenericPatternRef::Clear` for
    /// a clear string or `GenericPatternRef::Enc` for an encrypted string.
    ///
    /// # Examples
    ///
    /// ```rust
    /// use tfhe::integer::{ClientKey, ServerKey};
    /// use tfhe::shortint::parameters::PARAM_MESSAGE_2_CARRY_2_KS_PBS_TUNIFORM_2M128;
    /// use tfhe::strings::ciphertext::{FheString, GenericPattern};
    ///
    /// let ck = ClientKey::new(PARAM_MESSAGE_2_CARRY_2_KS_PBS_TUNIFORM_2M128);
    /// let sk = ServerKey::new_radix_server_key(&ck);
    /// let ck = tfhe::strings::ClientKey::new(ck);
    /// let sk = tfhe::strings::ServerKey::new(sk);
    /// let (s1, s2) = ("hello", "hello");
    ///
    /// let enc_s1 = FheString::new(&ck, s1, None);
    /// let enc_s2 = GenericPattern::Enc(FheString::new(&ck, s2, None));
    ///
    /// let result = sk.eq(&enc_s1, enc_s2.as_ref());
    /// let are_equal = ck.inner().decrypt_bool(&result);
    ///
    /// assert!(are_equal);
    /// ```
    pub fn eq(&self, lhs: &FheString, rhs: GenericPatternRef<'_>) -> BooleanBlock {
        let sk = self.inner();

        let early_return = match rhs {
            GenericPatternRef::Clear(rhs) => {
                self.eq_length_checks(lhs, &FheString::trivial(self, rhs.str()))
            }
            GenericPatternRef::Enc(rhs) => self.eq_length_checks(lhs, rhs),
        };

        if let Some(val) = early_return {
            return val;
        }

        let mut lhs_uint = lhs.to_uint();
        match rhs {
            GenericPatternRef::Clear(rhs) => {
                let rhs_clear_uint = self.pad_cipher_and_cleartext_lsb(&mut lhs_uint, rhs.str());

                sk.scalar_eq_parallelized(&lhs_uint, rhs_clear_uint)
            }
            GenericPatternRef::Enc(rhs) => {
                let mut rhs_uint = rhs.to_uint();

                self.pad_ciphertexts_lsb(&mut lhs_uint, &mut rhs_uint);

                sk.eq_parallelized(&lhs_uint, &rhs_uint)
            }
        }
    }

    /// Returns `true` if an encrypted string and a pattern (either encrypted or clear) are not
    /// equal.
    ///
    /// Returns `false` if they are equal.
    ///
    /// The pattern for comparison (`rhs`) can be specified as either `GenericPatternRef::Clear` for
    /// a clear string or `GenericPatternRef::Enc` for an encrypted string.
    ///
    /// # Examples
    ///
    /// ```rust
    /// use tfhe::integer::{ClientKey, ServerKey};
    /// use tfhe::shortint::parameters::PARAM_MESSAGE_2_CARRY_2_KS_PBS_TUNIFORM_2M128;
    /// use tfhe::strings::ciphertext::{FheString, GenericPattern};
    ///
    /// let ck = ClientKey::new(PARAM_MESSAGE_2_CARRY_2_KS_PBS_TUNIFORM_2M128);
    /// let sk = ServerKey::new_radix_server_key(&ck);
    /// let ck = tfhe::strings::ClientKey::new(ck);
    /// let sk = tfhe::strings::ServerKey::new(sk);
    /// let (s1, s2) = ("hello", "world");
    ///
    /// let enc_s1 = FheString::new(&ck, s1, None);
    /// let enc_s2 = GenericPattern::Enc(FheString::new(&ck, s2, None));
    ///
    /// let result = sk.ne(&enc_s1, enc_s2.as_ref());
    /// let are_not_equal = ck.inner().decrypt_bool(&result);
    ///
    /// assert!(are_not_equal);
    /// ```
    pub fn ne(&self, lhs: &FheString, rhs: GenericPatternRef<'_>) -> BooleanBlock {
        let sk = self.inner();

        let eq = self.eq(lhs, rhs);

        sk.boolean_bitnot(&eq)
    }

    /// Returns `true` if the first encrypted string is less than the second encrypted string.
    ///
    /// Returns `false` otherwise.
    ///
    /// # Examples
    ///
    /// ```rust
    /// use tfhe::integer::{ClientKey, ServerKey};
    /// use tfhe::shortint::parameters::PARAM_MESSAGE_2_CARRY_2_KS_PBS_TUNIFORM_2M128;
    /// use tfhe::strings::ciphertext::{FheString, GenericPattern};
    ///
    /// let ck = ClientKey::new(PARAM_MESSAGE_2_CARRY_2_KS_PBS_TUNIFORM_2M128);
    /// let sk = ServerKey::new_radix_server_key(&ck);
    /// let ck = tfhe::strings::ClientKey::new(ck);
    /// let sk = tfhe::strings::ServerKey::new(sk);
    /// let (s1, s2) = ("apple", "banana");
    ///
    /// let enc_s1 = FheString::new(&ck, s1, None);
    /// let enc_s2 = GenericPattern::Enc(FheString::new(&ck, s2, None));
    ///
    /// let result = sk.lt(&enc_s1, enc_s2.as_ref());
    /// let is_lt = ck.inner().decrypt_bool(&result);
    ///
    /// assert!(is_lt); // "apple" is less than "banana"
    /// ```
    pub fn lt(&self, lhs: &FheString, rhs: GenericPatternRef<'_>) -> BooleanBlock {
        let sk = self.inner();

        let mut lhs_uint = lhs.to_uint();

        let mut rhs_uint = match rhs {
            GenericPatternRef::Clear(rhs) => FheString::trivial(self, rhs.str()).to_uint(),
            GenericPatternRef::Enc(rhs) => rhs.to_uint(),
        };

        self.pad_ciphertexts_lsb(&mut lhs_uint, &mut rhs_uint);

        sk.lt_parallelized(&lhs_uint, &rhs_uint)
    }

    /// Returns `true` if the first encrypted string is greater than the second encrypted string.
    ///
    /// Returns `false` otherwise.
    ///
    /// # Examples
    ///
    /// ```rust
    /// use tfhe::integer::{ClientKey, ServerKey};
    /// use tfhe::shortint::parameters::PARAM_MESSAGE_2_CARRY_2_KS_PBS_TUNIFORM_2M128;
    /// use tfhe::strings::ciphertext::{FheString, GenericPattern};
    ///
    /// let ck = ClientKey::new(PARAM_MESSAGE_2_CARRY_2_KS_PBS_TUNIFORM_2M128);
    /// let sk = ServerKey::new_radix_server_key(&ck);
    /// let ck = tfhe::strings::ClientKey::new(ck);
    /// let sk = tfhe::strings::ServerKey::new(sk);
    /// let (s1, s2) = ("banana", "apple");
    ///
    /// let enc_s1 = FheString::new(&ck, s1, None);
    /// let enc_s2 = GenericPattern::Enc(FheString::new(&ck, s2, None));
    ///
    /// let result = sk.gt(&enc_s1, enc_s2.as_ref());
    /// let is_gt = ck.inner().decrypt_bool(&result);
    ///
    /// assert!(is_gt); // "banana" is greater than "apple"
    /// ```
    pub fn gt(&self, lhs: &FheString, rhs: GenericPatternRef<'_>) -> BooleanBlock {
        let sk = self.inner();

        let mut lhs_uint = lhs.to_uint();
        let mut rhs_uint = match rhs {
            GenericPatternRef::Clear(rhs) => FheString::trivial(self, rhs.str()).to_uint(),
            GenericPatternRef::Enc(rhs) => rhs.to_uint(),
        };

        self.pad_ciphertexts_lsb(&mut lhs_uint, &mut rhs_uint);

        sk.gt_parallelized(&lhs_uint, &rhs_uint)
    }

    /// Returns `true` if the first encrypted string is less than or equal to the second encrypted
    /// string.
    ///
    /// Returns `false` otherwise.
    ///
    /// # Examples
    ///
    /// ```rust
    /// use tfhe::integer::{ClientKey, ServerKey};
    /// use tfhe::shortint::parameters::PARAM_MESSAGE_2_CARRY_2_KS_PBS_TUNIFORM_2M128;
    /// use tfhe::strings::ciphertext::{FheString, GenericPattern};
    ///
    /// let ck = ClientKey::new(PARAM_MESSAGE_2_CARRY_2_KS_PBS_TUNIFORM_2M128);
    /// let sk = ServerKey::new_radix_server_key(&ck);
    /// let ck = tfhe::strings::ClientKey::new(ck);
    /// let sk = tfhe::strings::ServerKey::new(sk);
    /// let (s1, s2) = ("apple", "banana");
    ///
    /// let enc_s1 = FheString::new(&ck, s1, None);
    /// let enc_s2 = GenericPattern::Enc(FheString::new(&ck, s2, None));
    ///
    /// let result = sk.le(&enc_s1, enc_s2.as_ref());
    /// let is_le = ck.inner().decrypt_bool(&result);
    ///
    /// assert!(is_le); // "apple" is less than or equal to "banana"
    /// ```
    pub fn le(&self, lhs: &FheString, rhs: GenericPatternRef<'_>) -> BooleanBlock {
        let sk = self.inner();

        let mut lhs_uint = lhs.to_uint();
        let mut rhs_uint = match rhs {
            GenericPatternRef::Clear(rhs) => FheString::trivial(self, rhs.str()).to_uint(),
            GenericPatternRef::Enc(rhs) => rhs.to_uint(),
        };
        self.pad_ciphertexts_lsb(&mut lhs_uint, &mut rhs_uint);

        sk.le_parallelized(&lhs_uint, &rhs_uint)
    }

    /// Returns `true` if the first encrypted string is greater than or equal to the second
    /// encrypted string.
    ///
    /// Returns `false` otherwise.
    ///
    /// # Examples
    ///
    /// ```rust
    /// use tfhe::integer::{ClientKey, ServerKey};
    /// use tfhe::shortint::parameters::PARAM_MESSAGE_2_CARRY_2_KS_PBS_TUNIFORM_2M128;
    /// use tfhe::strings::ciphertext::{FheString, GenericPattern};
    ///
    /// let ck = ClientKey::new(PARAM_MESSAGE_2_CARRY_2_KS_PBS_TUNIFORM_2M128);
    /// let sk = ServerKey::new_radix_server_key(&ck);
    /// let ck = tfhe::strings::ClientKey::new(ck);
    /// let sk = tfhe::strings::ServerKey::new(sk);
    /// let (s1, s2) = ("banana", "apple");
    ///
    /// let enc_s1 = FheString::new(&ck, s1, None);
    /// let enc_s2 = GenericPattern::Enc(FheString::new(&ck, s2, None));
    ///
    /// let result = sk.ge(&enc_s1, enc_s2.as_ref());
    /// let is_ge = ck.inner().decrypt_bool(&result);
    ///
    /// assert!(is_ge); // "banana" is greater than or equal to "apple"
    /// ```
    pub fn ge(&self, lhs: &FheString, rhs: GenericPatternRef<'_>) -> BooleanBlock {
        let sk = self.inner();

        let mut lhs_uint = lhs.to_uint();
        let mut rhs_uint = match rhs {
            GenericPatternRef::Clear(rhs) => FheString::trivial(self, rhs.str()).to_uint(),
            GenericPatternRef::Enc(rhs) => rhs.to_uint(),
        };

        self.pad_ciphertexts_lsb(&mut lhs_uint, &mut rhs_uint);

        sk.ge_parallelized(&lhs_uint, &rhs_uint)
    }

    /// Returns `true` if an encrypted string and a pattern (either encrypted or clear) are equal,
    /// ignoring case differences.
    ///
    /// Returns `false` if they are not equal.
    ///
    /// The pattern for comparison (`rhs`) can be specified as either `GenericPatternRef::Clear` for
    /// a clear string or `GenericPatternRef::Enc` for an encrypted string.
    ///
    /// # Examples
    ///
    /// ```rust
    /// use tfhe::integer::{ClientKey, ServerKey};
    /// use tfhe::shortint::parameters::PARAM_MESSAGE_2_CARRY_2_KS_PBS_TUNIFORM_2M128;
    /// use tfhe::strings::ciphertext::{FheString, GenericPattern};
    ///
    /// let ck = ClientKey::new(PARAM_MESSAGE_2_CARRY_2_KS_PBS_TUNIFORM_2M128);
    /// let sk = ServerKey::new_radix_server_key(&ck);
    /// let ck = tfhe::strings::ClientKey::new(ck);
    /// let sk = tfhe::strings::ServerKey::new(sk);
    /// let (s1, s2) = ("Hello", "hello");
    ///
    /// let enc_s1 = FheString::new(&ck, s1, None);
    /// let enc_s2 = GenericPattern::Enc(FheString::new(&ck, s2, None));
    ///
    /// let result = sk.eq_ignore_case(&enc_s1, enc_s2.as_ref());
    /// let are_equal = ck.inner().decrypt_bool(&result);
    ///
    /// assert!(are_equal);
    /// ```
    pub fn eq_ignore_case(&self, lhs: &FheString, rhs: GenericPatternRef<'_>) -> BooleanBlock {
        let (lhs, rhs) = rayon::join(
            || self.to_lowercase(lhs),
            || match rhs {
                GenericPatternRef::Clear(rhs) => {
                    GenericPattern::Clear(ClearString::new(rhs.str().to_lowercase()))
                }
                GenericPatternRef::Enc(rhs) => GenericPattern::Enc(self.to_lowercase(rhs)),
            },
        );

        self.eq(&lhs, rhs.as_ref())
    }
}