pallet_insecure_randomness_collective_flip/
lib.rs1#![cfg_attr(not(feature = "std"), no_std)]
70
71use safe_mix::TripletMix;
72
73use codec::Encode;
74use frame::{prelude::*, traits::Randomness};
75
76const RANDOM_MATERIAL_LEN: u32 = 81;
77
78fn block_number_to_index<T: Config>(block_number: BlockNumberFor<T>) -> usize {
79 let index = (block_number - 1u32.into()) % RANDOM_MATERIAL_LEN.into();
81 index.try_into().ok().expect("Something % 81 is always smaller than usize; qed")
82}
83
84pub use pallet::*;
85
86#[frame::pallet]
87pub mod pallet {
88 use super::*;
89
90 #[pallet::pallet]
91 pub struct Pallet<T>(_);
92
93 #[pallet::config]
94 pub trait Config: frame_system::Config {}
95
96 #[pallet::hooks]
97 impl<T: Config> Hooks<BlockNumberFor<T>> for Pallet<T> {
98 fn on_initialize(block_number: BlockNumberFor<T>) -> Weight {
99 let parent_hash = frame_system::Pallet::<T>::parent_hash();
100
101 RandomMaterial::<T>::mutate(|ref mut values| {
102 if values.try_push(parent_hash).is_err() {
103 let index = block_number_to_index::<T>(block_number);
104 values[index] = parent_hash;
105 }
106 });
107
108 T::DbWeight::get().reads_writes(1, 1)
109 }
110 }
111
112 #[pallet::storage]
116 pub type RandomMaterial<T: Config> =
117 StorageValue<_, BoundedVec<T::Hash, ConstU32<RANDOM_MATERIAL_LEN>>, ValueQuery>;
118
119 impl<T: Config> Pallet<T> {
120 pub fn random_material() -> BoundedVec<T::Hash, ConstU32<RANDOM_MATERIAL_LEN>> {
122 RandomMaterial::<T>::get()
123 }
124 }
125}
126
127impl<T: Config> Randomness<T::Hash, BlockNumberFor<T>> for Pallet<T> {
128 fn random(subject: &[u8]) -> (T::Hash, BlockNumberFor<T>) {
139 let block_number = frame_system::Pallet::<T>::block_number();
140 let index = block_number_to_index::<T>(block_number);
141
142 let hash_series = RandomMaterial::<T>::get();
143 let seed = if !hash_series.is_empty() {
144 hash_series
146 .iter()
147 .cycle()
148 .skip(index)
149 .take(RANDOM_MATERIAL_LEN as usize)
150 .enumerate()
151 .map(|(i, h)| (i as i8, subject, h).using_encoded(T::Hashing::hash))
152 .triplet_mix()
153 } else {
154 T::Hash::default()
155 };
156
157 (seed, block_number.saturating_sub(RANDOM_MATERIAL_LEN.into()))
158 }
159}
160
161#[cfg(test)]
162mod tests {
163 use super::*;
164 use crate as pallet_insecure_randomness_collective_flip;
165 use frame::{
166 testing_prelude::{frame_system::limits, *},
167 traits::Header as _,
168 };
169
170 type Block = frame_system::mocking::MockBlock<Test>;
171
172 construct_runtime!(
173 pub enum Test
174 {
175 System: frame_system,
176 CollectiveFlip: pallet_insecure_randomness_collective_flip,
177 }
178 );
179
180 parameter_types! {
181 pub BlockLength: limits::BlockLength = limits::BlockLength::builder()
182 .max_length(2 * 1024)
183 .build();
184 }
185
186 #[derive_impl(frame_system::config_preludes::TestDefaultConfig)]
187 impl frame_system::Config for Test {
188 type Block = Block;
189 }
190
191 impl pallet_insecure_randomness_collective_flip::Config for Test {}
192
193 fn new_test_ext() -> TestExternalities {
194 let t = frame_system::GenesisConfig::<Test>::default().build_storage().unwrap();
195 t.into()
196 }
197
198 #[test]
199 fn test_block_number_to_index() {
200 for i in 1..1000 {
201 assert_eq!((i - 1) as usize % 81, block_number_to_index::<Test>(i));
202 }
203 }
204
205 fn setup_blocks(blocks: u64) {
206 let mut parent_hash = System::parent_hash();
207
208 for i in 1..(blocks + 1) {
209 System::reset_events();
210 System::initialize(&i, &parent_hash, &Default::default());
211 CollectiveFlip::on_initialize(i);
212
213 let header = System::finalize();
214 parent_hash = header.hash();
215 System::set_block_number(*header.number());
216 }
217 }
218
219 #[test]
220 fn test_random_material_partial() {
221 new_test_ext().execute_with(|| {
222 let genesis_hash = System::parent_hash();
223
224 setup_blocks(38);
225
226 let random_material = RandomMaterial::<Test>::get();
227
228 assert_eq!(random_material.len(), 38);
229 assert_eq!(random_material[0], genesis_hash);
230 });
231 }
232
233 #[test]
234 fn test_random_material_filled() {
235 new_test_ext().execute_with(|| {
236 let genesis_hash = System::parent_hash();
237
238 setup_blocks(81);
239
240 let random_material = RandomMaterial::<Test>::get();
241
242 assert_eq!(random_material.len(), 81);
243 assert_ne!(random_material[0], random_material[1]);
244 assert_eq!(random_material[0], genesis_hash);
245 });
246 }
247
248 #[test]
249 fn test_random_material_filled_twice() {
250 new_test_ext().execute_with(|| {
251 let genesis_hash = System::parent_hash();
252
253 setup_blocks(162);
254
255 let random_material = RandomMaterial::<Test>::get();
256
257 assert_eq!(random_material.len(), 81);
258 assert_ne!(random_material[0], random_material[1]);
259 assert_ne!(random_material[0], genesis_hash);
260 });
261 }
262
263 #[test]
264 fn test_random() {
265 new_test_ext().execute_with(|| {
266 setup_blocks(162);
267
268 assert_eq!(System::block_number(), 162);
269 assert_eq!(CollectiveFlip::random_seed(), CollectiveFlip::random_seed());
270 assert_ne!(CollectiveFlip::random(b"random_1"), CollectiveFlip::random(b"random_2"));
271
272 let (random, known_since) = CollectiveFlip::random_seed();
273
274 assert_eq!(known_since, 162 - RANDOM_MATERIAL_LEN as u64);
275 assert_ne!(random, H256::zero());
276 assert!(!RandomMaterial::<Test>::get().contains(&random));
277 });
278 }
279}