// This file is part of Substrate. // Copyright (C) Parity Technologies (UK) Ltd. // SPDX-License-Identifier: Apache-2.0 // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. //! Tests for the glutton pallet. use super::{mock::*, *}; use frame_support::{assert_err, assert_noop, assert_ok, weights::constants::*}; use sp_runtime::{traits::One, Perbill}; const CALIBRATION_ERROR: &'static str = "Weight calibration failed. Please re-run the benchmarks on the same hardware."; #[test] fn initialize_pallet_works() { new_test_ext().execute_with(|| { assert_eq!(TrashData::::get(0), None); assert_noop!( Glutton::initialize_pallet(RuntimeOrigin::signed(1), 3, None), DispatchError::BadOrigin ); assert_noop!( Glutton::initialize_pallet(RuntimeOrigin::none(), 3, None), DispatchError::BadOrigin ); assert_ok!(Glutton::initialize_pallet(RuntimeOrigin::root(), 2, None)); System::assert_last_event(Event::PalletInitialized { reinit: false }.into()); assert_err!( Glutton::initialize_pallet(RuntimeOrigin::root(), 2, None), Error::::AlreadyInitialized ); assert_eq!(TrashData::::get(0), Some(Pallet::::gen_value(0))); assert_eq!(TrashData::::get(1), Some(Pallet::::gen_value(1))); assert_eq!(TrashData::::get(2), None); assert_eq!(TrashDataCount::::get(), 2); assert_ok!(Glutton::initialize_pallet(RuntimeOrigin::root(), 20, Some(2))); assert_eq!(TrashDataCount::::get(), 20); assert_eq!(TrashData::::iter_keys().count(), 20); }); } #[test] fn expand_and_shrink_trash_data_works() { new_test_ext().execute_with(|| { assert_eq!(TrashDataCount::::get(), 0); assert_ok!(Glutton::initialize_pallet(RuntimeOrigin::root(), 5000, None)); assert_eq!(TrashDataCount::::get(), 5000); assert_eq!(TrashData::::iter_keys().count(), 5000); assert_ok!(Glutton::initialize_pallet(RuntimeOrigin::root(), 8000, Some(5000))); assert_eq!(TrashDataCount::::get(), 8000); assert_eq!(TrashData::::iter_keys().count(), 8000); assert_ok!(Glutton::initialize_pallet(RuntimeOrigin::root(), 6000, Some(8000))); assert_eq!(TrashDataCount::::get(), 6000); assert_eq!(TrashData::::iter_keys().count(), 6000); assert_noop!( Glutton::initialize_pallet(RuntimeOrigin::root(), 0, None), Error::::AlreadyInitialized ); assert_ok!(Glutton::initialize_pallet(RuntimeOrigin::root(), 0, Some(6000))); assert_eq!(TrashDataCount::::get(), 0); assert_eq!(TrashData::::iter_keys().count(), 0); }); } #[test] fn setting_compute_works() { new_test_ext().execute_with(|| { assert_eq!(Compute::::get(), Zero::zero()); assert_ok!(Glutton::set_compute(RuntimeOrigin::root(), FixedU64::from_float(0.3))); assert_eq!(Compute::::get(), FixedU64::from_float(0.3)); System::assert_last_event( Event::ComputationLimitSet { compute: FixedU64::from_float(0.3) }.into(), ); assert_noop!( Glutton::set_compute(RuntimeOrigin::signed(1), FixedU64::from_float(0.5)), DispatchError::BadOrigin ); assert_noop!( Glutton::set_compute(RuntimeOrigin::none(), FixedU64::from_float(0.5)), DispatchError::BadOrigin ); }); } #[test] fn setting_compute_respects_limit() { new_test_ext().execute_with(|| { // < 1000% is fine assert_ok!(Glutton::set_compute(RuntimeOrigin::root(), FixedU64::from_float(9.99)),); // == 1000% is fine assert_ok!(Glutton::set_compute(RuntimeOrigin::root(), FixedU64::from_u32(10)),); // > 1000% is not assert_noop!( Glutton::set_compute(RuntimeOrigin::root(), FixedU64::from_float(10.01)), Error::::InsaneLimit ); }); } #[test] fn setting_storage_works() { new_test_ext().execute_with(|| { assert!(Storage::::get().is_zero()); assert_ok!(Glutton::set_storage(RuntimeOrigin::root(), FixedU64::from_float(0.3))); assert_eq!(Storage::::get(), FixedU64::from_float(0.3)); System::assert_last_event( Event::StorageLimitSet { storage: FixedU64::from_float(0.3) }.into(), ); assert_noop!( Glutton::set_storage(RuntimeOrigin::signed(1), FixedU64::from_float(0.5)), DispatchError::BadOrigin ); assert_noop!( Glutton::set_storage(RuntimeOrigin::none(), FixedU64::from_float(0.5)), DispatchError::BadOrigin ); }); } #[test] fn setting_storage_respects_limit() { new_test_ext().execute_with(|| { // < 1000% is fine assert_ok!(Glutton::set_storage(RuntimeOrigin::root(), FixedU64::from_float(9.99)),); // == 1000% is fine assert_ok!(Glutton::set_storage(RuntimeOrigin::root(), FixedU64::from_u32(10)),); // > 1000% is not assert_noop!( Glutton::set_storage(RuntimeOrigin::root(), FixedU64::from_float(10.01)), Error::::InsaneLimit ); }); } #[test] fn on_idle_works() { new_test_ext().execute_with(|| { set_limits(One::one(), One::one()); Glutton::on_idle(1, Weight::from_parts(20_000_000, 0)); }); } /// Check that the expected is close enough to the consumed weight. #[test] fn on_idle_weight_high_proof_is_close_enough_works() { new_test_ext().execute_with(|| { set_limits(One::one(), One::one()); let should = Weight::from_parts(WEIGHT_REF_TIME_PER_SECOND, WEIGHT_PROOF_SIZE_PER_MB * 5); let got = Glutton::on_idle(1, should); assert!(got.all_lte(should), "Consumed too much weight"); let ratio = Perbill::from_rational(got.proof_size(), should.proof_size()); assert!( ratio >= Perbill::from_percent(99), "Too few proof size consumed, was only {ratio:?} of expected", ); let ratio = Perbill::from_rational(got.ref_time(), should.ref_time()); assert!( ratio >= Perbill::from_percent(99), "Too few ref time consumed, was only {ratio:?} of expected", ); }); } #[test] fn on_idle_weight_low_proof_is_close_enough_works() { new_test_ext().execute_with(|| { set_limits(One::one(), One::one()); let should = Weight::from_parts(WEIGHT_REF_TIME_PER_SECOND, WEIGHT_PROOF_SIZE_PER_KB * 20); let got = Glutton::on_idle(1, should); assert!(got.all_lte(should), "Consumed too much weight"); let ratio = Perbill::from_rational(got.proof_size(), should.proof_size()); // Just a sanity check here for > 0 assert!( ratio >= Perbill::from_percent(50), "Too few proof size consumed, was only {ratio:?} of expected", ); let ratio = Perbill::from_rational(got.ref_time(), should.ref_time()); assert!( ratio >= Perbill::from_percent(99), "Too few ref time consumed, was only {ratio:?} of expected", ); }); } #[test] fn on_idle_weight_over_unity_is_close_enough_works() { new_test_ext().execute_with(|| { // Para blocks get ~500ms compute and ~5MB proof size. let max_block = Weight::from_parts(500 * WEIGHT_REF_TIME_PER_MILLIS, 5 * WEIGHT_PROOF_SIZE_PER_MB); // But now we tell it to consume more than that. set_limits(1.75, 1.5); let want = Weight::from_parts( (1.75 * max_block.ref_time() as f64) as u64, (1.5 * max_block.proof_size() as f64) as u64, ); let consumed = Glutton::on_idle(1, max_block); assert!(consumed.all_gt(max_block), "Must consume more than the block limit"); assert!(consumed.all_lte(want), "Consumed more than the requested weight"); let ratio = Perbill::from_rational(consumed.proof_size(), want.proof_size()); assert!( ratio >= Perbill::from_percent(99), "Too few proof size consumed, was only {ratio:?} of expected", ); let ratio = Perbill::from_rational(consumed.ref_time(), want.ref_time()); assert!( ratio >= Perbill::from_percent(99), "Too few ref time consumed, was only {ratio:?} of expected", ); }); } #[test] fn waste_at_most_ref_time_weight_close_enough() { new_test_ext().execute_with(|| { let mut meter = WeightMeter::with_limit(Weight::from_parts(WEIGHT_REF_TIME_PER_SECOND, u64::MAX)); // Over-spending fails defensively. Glutton::waste_at_most_ref_time(&mut meter); // We require it to be under-spend by at most 1%. assert!( meter.consumed_ratio() >= Perbill::from_percent(99), "{CALIBRATION_ERROR}\nConsumed too few: {:?}", meter.consumed_ratio() ); }); } #[test] fn waste_at_most_proof_size_weight_close_enough() { new_test_ext().execute_with(|| { let mut meter = WeightMeter::with_limit(Weight::from_parts(u64::MAX, WEIGHT_PROOF_SIZE_PER_MB * 5)); // Over-spending fails defensively. Glutton::waste_at_most_proof_size(&mut meter); // We require it to be under-spend by at most 1%. assert!( meter.consumed_ratio() >= Perbill::from_percent(99), "{CALIBRATION_ERROR}\nConsumed too few: {:?}", meter.consumed_ratio() ); }); } #[test] fn gen_value_works() { let g0 = Pallet::::gen_value(0); let g1 = Pallet::::gen_value(1); assert_eq!(g0.len(), VALUE_SIZE); assert_ne!(g0, g1, "Is distinct"); assert_ne!(g0, [0; VALUE_SIZE], "Is not zero"); assert_eq!(g0, Pallet::::gen_value(0), "Is deterministic"); }