use crate::{
error::MathError,
integer_mod_q::{Modulus, Zq},
rational::Q,
utils::sample::discrete_gauss::{DiscreteGaussianIntegerSampler, LookupTableSetting, TAILCUT},
};
impl Zq {
pub fn sample_discrete_gauss(
modulus: impl Into<Modulus>,
center: impl Into<Q>,
s: impl Into<Q>,
) -> Result<Self, MathError> {
let modulus: Modulus = modulus.into();
let center: Q = center.into();
let s: Q = s.into();
let mut dgis = DiscreteGaussianIntegerSampler::init(
¢er,
&s,
unsafe { TAILCUT },
LookupTableSetting::NoLookup,
)?;
let sample = dgis.sample_z();
Ok(Zq::from((&sample, &modulus)))
}
}
#[cfg(test)]
mod test_sample_discrete_gauss {
use crate::{integer::Z, integer_mod_q::Zq, rational::Q};
#[test]
fn availability() {
let z = Z::from(2);
let q = Q::from(2);
let _ = Zq::sample_discrete_gauss(17u64, 7u8, 1u16);
let _ = Zq::sample_discrete_gauss(17u16, 7u16, 1u8);
let _ = Zq::sample_discrete_gauss(17u8, 7u32, 1u32);
let _ = Zq::sample_discrete_gauss(17u32, 7u64, 1u64);
let _ = Zq::sample_discrete_gauss(17i16, 7i8, 1i64);
let _ = Zq::sample_discrete_gauss(17i8, 7i16, 1i32);
let _ = Zq::sample_discrete_gauss(17i64, 7i32, 1i16);
let _ = Zq::sample_discrete_gauss(17i32, 7i64, 1i8);
let _ = Zq::sample_discrete_gauss(17, q.clone(), 1i64);
let _ = Zq::sample_discrete_gauss(z.clone(), 0i8, z.clone());
let _ = Zq::sample_discrete_gauss(17, z.clone(), q.clone());
let _ = Zq::sample_discrete_gauss(17, 1f32, 1f64);
let _ = Zq::sample_discrete_gauss(17, 1f64, 1f32);
let _ = Zq::sample_discrete_gauss(17i64, 7i32, 1i16);
let _ = Zq::sample_discrete_gauss(17i32, 7i64, 1i8);
let _ = Zq::sample_discrete_gauss(17, &q, 1i64);
let _ = Zq::sample_discrete_gauss(&z, 0i8, &z);
let _ = Zq::sample_discrete_gauss(17, &z, &q);
}
#[test]
fn distribution() {
let mut counts = [0; 20];
for _ in 0..200 {
let sample = Zq::sample_discrete_gauss(20, 0, 2).unwrap();
let sample_int = i64::try_from(&sample.get_representative_least_nonnegative_residue())
.unwrap() as usize;
counts[sample_int] += 1;
}
let expl_text = String::from("This test can fail with probability close to 0.
It fails if the sampled occurrences do not look like a typical discrete Gaussian random distribution.
If this happens, rerun the tests several times and check whether this issue comes up again.");
assert!(counts[0] > 70, "{expl_text}");
assert!(counts[0] < 130, "{expl_text}");
assert!(counts[19] > 20, "{expl_text}");
assert!(counts[19] < 70, "{expl_text}");
assert!(counts[1] > 20, "{expl_text}");
assert!(counts[1] < 70, "{expl_text}");
assert!(counts[18] < 20, "{expl_text}");
assert!(counts[2] < 20, "{expl_text}");
for count in counts.iter().take(18).skip(3) {
assert!(count < &10, "{expl_text}");
}
}
}