use crate::{
error::MathError,
integer::Z,
integer_mod_q::{Modulus, PolyOverZq},
traits::SetCoefficient,
utils::{index::evaluate_index, sample::uniform::UniformIntegerSampler},
};
use std::fmt::Display;
impl PolyOverZq {
pub fn sample_uniform(
max_degree: impl TryInto<i64> + Display + Copy,
modulus: impl Into<Z>,
) -> Result<Self, MathError> {
let max_degree = evaluate_index(max_degree)?;
let interval_size = modulus.into();
let modulus = Modulus::from(&interval_size);
let mut poly_zq = PolyOverZq::from(&modulus);
let mut uis = UniformIntegerSampler::init(&interval_size)?;
for index in 0..=max_degree {
let sample = uis.sample();
unsafe { poly_zq.set_coeff_unchecked(index, sample) };
}
Ok(poly_zq)
}
}
#[cfg(test)]
mod test_sample_uniform {
use crate::{
integer::Z,
integer_mod_q::{Modulus, PolyOverZq},
traits::GetCoefficient,
};
#[test]
fn boundaries_kept_small() {
let modulus = Z::from(17);
let poly_zq = PolyOverZq::sample_uniform(32, &modulus).unwrap();
for i in 0..32 {
let sample: Z = poly_zq.get_coeff(i).unwrap();
assert!(Z::ZERO <= sample);
assert!(sample < modulus);
}
}
#[test]
fn boundaries_kept_large() {
let modulus = Z::from(i64::MAX);
let poly_zq = PolyOverZq::sample_uniform(256, &modulus).unwrap();
for i in 0..256 {
let sample: Z = poly_zq.get_coeff(i).unwrap();
assert!(Z::ZERO <= sample);
assert!(sample < modulus);
}
}
#[test]
fn nr_coeffs() {
let degrees = [1, 3, 7, 15, 32, 120];
for degree in degrees {
let res = PolyOverZq::sample_uniform(degree, u64::MAX).unwrap();
assert_eq!(
degree,
res.get_degree(),
"Could fail with probability 1/{}.",
u64::MAX
);
}
}
#[test]
#[should_panic]
fn invalid_modulus_negative() {
let _ = PolyOverZq::sample_uniform(1, i64::MIN);
}
#[test]
#[should_panic]
fn invalid_modulus_one() {
let _ = PolyOverZq::sample_uniform(1, 1);
}
#[test]
fn invalid_max_degree() {
let modulus = Z::from(15);
let res_0 = PolyOverZq::sample_uniform(-1, &modulus);
let res_1 = PolyOverZq::sample_uniform(i64::MIN, &modulus);
assert!(res_0.is_err());
assert!(res_1.is_err());
}
#[test]
fn availability() {
let modulus = Modulus::from(10);
let z = Z::from(10);
let _ = PolyOverZq::sample_uniform(1u64, 10u16).unwrap();
let _ = PolyOverZq::sample_uniform(1i64, 10u32).unwrap();
let _ = PolyOverZq::sample_uniform(1u8, 10u64).unwrap();
let _ = PolyOverZq::sample_uniform(1u16, 10i8).unwrap();
let _ = PolyOverZq::sample_uniform(1u32, 10i16).unwrap();
let _ = PolyOverZq::sample_uniform(1i32, 10i32).unwrap();
let _ = PolyOverZq::sample_uniform(1i16, 10i64).unwrap();
let _ = PolyOverZq::sample_uniform(1i8, &z).unwrap();
let _ = PolyOverZq::sample_uniform(1, z).unwrap();
let _ = PolyOverZq::sample_uniform(1, &modulus).unwrap();
let _ = PolyOverZq::sample_uniform(1, modulus).unwrap();
}
}