qfall_math/integer/poly_over_z/sample/
uniform.rs1use crate::{
12 error::MathError,
13 integer::{PolyOverZ, Z},
14 traits::SetCoefficient,
15 utils::{index::evaluate_index, sample::uniform::UniformIntegerSampler},
16};
17use std::fmt::Display;
18
19impl PolyOverZ {
20 pub fn sample_uniform(
52 max_degree: impl TryInto<i64> + Display,
53 lower_bound: impl Into<Z>,
54 upper_bound: impl Into<Z>,
55 ) -> Result<Self, MathError> {
56 let max_degree = evaluate_index(max_degree)?;
57 let lower_bound: Z = lower_bound.into();
58 let upper_bound: Z = upper_bound.into();
59
60 let interval_size = &upper_bound - &lower_bound;
61 let mut uis = UniformIntegerSampler::init(&interval_size)?;
62
63 let mut poly_z = PolyOverZ::default();
64 for index in 0..=max_degree {
65 let sample = uis.sample();
66 unsafe { poly_z.set_coeff_unchecked(index, &lower_bound + sample) };
67 }
68 Ok(poly_z)
69 }
70}
71
72#[cfg(test)]
73mod test_sample_uniform {
74 use crate::{
75 integer::{PolyOverZ, Z},
76 integer_mod_q::Modulus,
77 traits::GetCoefficient,
78 };
79
80 #[test]
82 fn boundaries_kept_small() {
83 let lower_bound = Z::from(17);
84 let upper_bound = Z::from(32);
85 let poly_z = PolyOverZ::sample_uniform(32, &lower_bound, &upper_bound).unwrap();
86
87 for i in 0..32 {
88 let sample = poly_z.get_coeff(i).unwrap();
89 assert!(lower_bound <= sample);
90 assert!(sample < upper_bound);
91 }
92 }
93
94 #[test]
96 fn boundaries_kept_large() {
97 let lower_bound = Z::from(i64::MIN) - Z::from(u64::MAX);
98 let upper_bound = Z::from(i64::MIN);
99
100 let poly_z = PolyOverZ::sample_uniform(256, &lower_bound, &upper_bound).unwrap();
101
102 for i in 0..256 {
103 let sample = poly_z.get_coeff(i).unwrap();
104 assert!(lower_bound <= sample);
105 assert!(sample < upper_bound);
106 }
107 }
108
109 #[test]
111 fn nr_coeffs() {
112 let degrees = [1, 3, 7, 15, 32, 120];
113 for degree in degrees {
114 let res = PolyOverZ::sample_uniform(degree, 1, 15).unwrap();
115
116 assert_eq!(degree, res.get_degree());
117 }
118 }
119
120 #[test]
122 fn invalid_interval() {
123 let lb_0 = Z::from(i64::MIN);
124 let lb_1 = Z::from(i64::MIN);
125 let lb_2 = Z::ZERO;
126 let upper_bound = Z::from(i64::MIN);
127
128 let res_0 = PolyOverZ::sample_uniform(1, &lb_0, &upper_bound);
129 let res_1 = PolyOverZ::sample_uniform(1, &lb_1, &upper_bound);
130 let res_2 = PolyOverZ::sample_uniform(1, &lb_2, &upper_bound);
131
132 assert!(res_0.is_err());
133 assert!(res_1.is_err());
134 assert!(res_2.is_err());
135 }
136
137 #[test]
139 fn invalid_max_degree() {
140 let lower_bound = Z::from(0);
141 let upper_bound = Z::from(15);
142
143 let res_0 = PolyOverZ::sample_uniform(-1, &lower_bound, &upper_bound);
144 let res_1 = PolyOverZ::sample_uniform(i64::MIN, &lower_bound, &upper_bound);
145
146 assert!(res_0.is_err());
147 assert!(res_1.is_err());
148 }
149
150 #[test]
153 fn availability() {
154 let modulus = Modulus::from(7);
155 let z = Z::from(7);
156
157 let _ = PolyOverZ::sample_uniform(1u64, 0u16, 7u8);
158 let _ = PolyOverZ::sample_uniform(1i64, 0u32, 7u16);
159 let _ = PolyOverZ::sample_uniform(1u8, 0u64, 7u32);
160 let _ = PolyOverZ::sample_uniform(1u16, 0i8, 7u64);
161 let _ = PolyOverZ::sample_uniform(1u32, 0i16, 7i8);
162 let _ = PolyOverZ::sample_uniform(1i32, 0i32, 7i16);
163 let _ = PolyOverZ::sample_uniform(1i16, 0i64, 7i32);
164 let _ = PolyOverZ::sample_uniform(1i8, &Z::ZERO, 7i64);
165 let _ = PolyOverZ::sample_uniform(1, 0u8, &modulus);
166 let _ = PolyOverZ::sample_uniform(1, 0, &z);
167 }
168}