use crate::{integer::Z, integer_mod_q::Zq, utils::sample::uniform::UniformIntegerSampler};
impl Zq {
pub fn sample_uniform(modulus: impl Into<Z>) -> Self {
let modulus: Z = modulus.into();
let mut uis = UniformIntegerSampler::init(&modulus).unwrap();
let random = uis.sample();
Zq::from((random, modulus))
}
}
#[cfg(test)]
mod test_sample_uniform {
use crate::{
integer::Z,
integer_mod_q::{Modulus, Zq},
};
#[test]
fn boundaries_kept_small() {
let modulus = Z::from(17);
for _ in 0..32 {
let sample = Zq::sample_uniform(&modulus);
assert!(Z::ZERO <= sample.value);
assert!(sample.value < modulus);
}
}
#[test]
fn boundaries_kept_large() {
let modulus = Z::from(u64::MAX);
for _ in 0..256 {
let sample = Zq::sample_uniform(&modulus);
assert!(Z::ZERO <= sample.value);
assert!(sample.value < modulus);
}
}
#[test]
#[should_panic]
fn invalid_interval() {
let modulus = Z::ZERO;
let _ = Zq::sample_uniform(&modulus);
}
#[test]
fn availability() {
let modulus = Modulus::from(7);
let z = Z::from(7);
let _ = Zq::sample_uniform(7u8);
let _ = Zq::sample_uniform(7u16);
let _ = Zq::sample_uniform(7u32);
let _ = Zq::sample_uniform(7u64);
let _ = Zq::sample_uniform(7i8);
let _ = Zq::sample_uniform(7i16);
let _ = Zq::sample_uniform(7i32);
let _ = Zq::sample_uniform(7i64);
let _ = Zq::sample_uniform(&modulus);
let _ = Zq::sample_uniform(&z);
}
#[test]
fn uniformity() {
let modulus = Z::from(5);
let mut counts = [0; 5];
for _ in 0..1000 {
let sample_z = Zq::sample_uniform(&modulus);
let sample_int = i64::try_from(&sample_z.value).unwrap() as usize;
counts[sample_int] += 1;
}
for count in counts {
assert!(count > 150, "This test can fail with probability close to 0.
It fails if the sampled occurrences do not look like a typical uniform random distribution.
If this happens, rerun the tests several times and check whether this issue comes up again.");
}
}
}