use cfg_primitives::{
	Balance, BlockNumber, DAYS, HOURS, MINUTES, TRACK_INDEX_POOL_ADMIN, TRACK_INDEX_REF_CANCELLER,
	TRACK_INDEX_REF_KILLER, TRACK_INDEX_ROOT, TRACK_INDEX_TREASURER,
	TRACK_INDEX_WHITELISTED_CALLER,
};
use cfg_utils::math::{to_percent, to_ppm};
use pallet_referenda::Curve;
use crate::{constants::currency::AIR, RuntimeOrigin};
const SUP_ROOT: Curve =
	Curve::make_reciprocal(1, 168, to_percent(25), to_ppm(3000), to_percent(50));
const APP_ROOT: Curve = Curve::make_linear(7, 7, to_percent(50), to_percent(100));
const SUP_WHITELISTED: Curve =
	Curve::make_reciprocal(1, 84, to_ppm(37000), to_ppm(500), to_percent(50));
const APP_WHITELISTED: Curve = Curve::make_linear(84, 84, to_percent(50), to_percent(100));
const SUP_POOL_ADMIN: Curve =
	Curve::make_reciprocal(1, 84, to_ppm(142900), to_ppm(2400), to_percent(50));
const APP_POOL_ADMIN: Curve = Curve::make_linear(84, 84, to_percent(70), to_percent(100));
const SUP_TREASURER: Curve = Curve::make_linear(13, 14, to_percent(1), to_percent(50));
const APP_TREASURER: Curve = Curve::make_linear(14, 14, to_percent(70), to_percent(100));
const SUP_REF_CANCELLER: Curve =
	Curve::make_reciprocal(24, 84, to_ppm(8200), to_ppm(2400), to_percent(50));
const APP_REF_CANCELLER: Curve = Curve::make_linear(84, 84, to_percent(70), to_percent(100));
const SUP_REF_KILLER: Curve =
	Curve::make_reciprocal(24, 84, to_ppm(8200), to_ppm(2400), to_percent(50));
const APP_REF_KILLER: Curve = Curve::make_linear(84, 84, to_percent(70), to_percent(100));
const TRACKS_DATA: [(u16, pallet_referenda::TrackInfo<Balance, BlockNumber>); 6] = [
	(
		TRACK_INDEX_ROOT,
		pallet_referenda::TrackInfo {
			name: "root",
			max_deciding: 2,
			decision_deposit: 3_000_000 * AIR,
			prepare_period: 3 * HOURS,
			decision_period: 7 * DAYS,
			confirm_period: 6 * HOURS,
			min_enactment_period: 12 * HOURS,
			min_approval: APP_ROOT,
			min_support: SUP_ROOT,
		},
	),
	(
		TRACK_INDEX_WHITELISTED_CALLER,
		pallet_referenda::TrackInfo {
			name: "whitelisted_caller",
			max_deciding: 20,
			decision_deposit: 10_000 * AIR,
			prepare_period: 5 * MINUTES,
			decision_period: 84 * HOURS,
			confirm_period: 5 * MINUTES,
			min_enactment_period: 5 * MINUTES,
			min_approval: APP_WHITELISTED,
			min_support: SUP_WHITELISTED,
		},
	),
	(
		TRACK_INDEX_POOL_ADMIN,
		pallet_referenda::TrackInfo {
			name: "pool_admin",
			max_deciding: 5,
			decision_deposit: 20_000 * AIR,
			prepare_period: 30 * MINUTES,
			decision_period: 84 * HOURS,
			confirm_period: 30 * MINUTES,
			min_enactment_period: 30 * MINUTES,
			min_approval: APP_POOL_ADMIN,
			min_support: SUP_POOL_ADMIN,
		},
	),
	(
		TRACK_INDEX_TREASURER,
		pallet_referenda::TrackInfo {
			name: "treasurer",
			max_deciding: 2,
			decision_deposit: 150_000 * AIR,
			prepare_period: 6 * HOURS,
			decision_period: 14 * DAYS,
			confirm_period: 12 * HOURS,
			min_enactment_period: 12 * HOURS,
			min_approval: APP_TREASURER,
			min_support: SUP_TREASURER,
		},
	),
	(
		TRACK_INDEX_REF_CANCELLER,
		pallet_referenda::TrackInfo {
			name: "referendum_canceller",
			max_deciding: 20,
			decision_deposit: 600_000 * AIR,
			prepare_period: 30 * MINUTES,
			decision_period: 84 * HOURS,
			confirm_period: 30 * MINUTES,
			min_enactment_period: 5 * MINUTES,
			min_approval: APP_REF_CANCELLER,
			min_support: SUP_REF_CANCELLER,
		},
	),
	(
		TRACK_INDEX_REF_KILLER,
		pallet_referenda::TrackInfo {
			name: "referendum_killer",
			max_deciding: 20,
			decision_deposit: 1_000_000 * AIR,
			prepare_period: 30 * MINUTES,
			decision_period: 84 * HOURS,
			confirm_period: 30 * MINUTES,
			min_enactment_period: 5 * MINUTES,
			min_approval: APP_REF_KILLER,
			min_support: SUP_REF_KILLER,
		},
	),
];
pub struct TracksInfo;
impl pallet_referenda::TracksInfo<Balance, BlockNumber> for TracksInfo {
	type Id = u16;
	type RuntimeOrigin = <RuntimeOrigin as frame_support::traits::OriginTrait>::PalletsOrigin;
	fn tracks() -> &'static [(Self::Id, pallet_referenda::TrackInfo<Balance, BlockNumber>)] {
		&TRACKS_DATA[..]
	}
	fn track_for(id: &Self::RuntimeOrigin) -> Result<Self::Id, ()> {
		if let Ok(system_origin) = frame_system::RawOrigin::try_from(id.clone()) {
			match system_origin {
				frame_system::RawOrigin::Root => Ok(TRACK_INDEX_ROOT),
				_ => Err(()),
			}
		} else if let Ok(custom_origin) = runtime_common::origins::gov::Origin::try_from(id.clone())
		{
			match custom_origin {
				runtime_common::origins::gov::Origin::WhitelistedCaller => {
					Ok(TRACK_INDEX_WHITELISTED_CALLER)
				}
				runtime_common::origins::gov::Origin::PoolAdmin => Ok(TRACK_INDEX_POOL_ADMIN),
				runtime_common::origins::gov::Origin::Treasurer => Ok(TRACK_INDEX_TREASURER),
				runtime_common::origins::gov::Origin::ReferendumCanceller => {
					Ok(TRACK_INDEX_REF_CANCELLER)
				}
				runtime_common::origins::gov::Origin::ReferendumKiller => {
					Ok(TRACK_INDEX_REF_KILLER)
				}
			}
		} else {
			Err(())
		}
	}
}
pallet_referenda::impl_tracksinfo_get!(TracksInfo, Balance, BlockNumber);
#[cfg(test)]
mod tests {
	use sp_arithmetic::Perbill;
	use super::*;
	const DECIMAL_PRECISION: u32 = 4;
	const PRECISION: u32 = 10u32.pow(DECIMAL_PRECISION);
	fn round(x: Perbill) -> Perbill {
		Perbill::from_rational(x * PRECISION, PRECISION)
	}
	#[test]
	fn root_track() {
		const HOURS: [u32; 8] = [0, 1, 2, 3, 24, 72, 120, 168];
		let approval: [Perbill; 8] = [
			Perbill::from_rational(100, 100u32),
			Perbill::from_rational(997, 1000u32),
			Perbill::from_rational(994, 1000u32),
			Perbill::from_rational(9911, 10000u32),
			Perbill::from_rational(9286, 10000u32),
			Perbill::from_rational(7857, 10000u32),
			Perbill::from_rational(6429, 10000u32),
			Perbill::from_rational(50, 100u32),
		];
		let sup: [Perbill; 8] = [
			Perbill::from_rational(50, 100u32),
			Perbill::from_rational(25, 100u32),
			Perbill::from_rational(1667, 10000u32),
			Perbill::from_rational(125, 1000u32),
			Perbill::from_rational(2, 100u32),
			Perbill::from_rational(69, 10000u32),
			Perbill::from_rational(42, 10000u32),
			Perbill::from_rational(3, 1000u32),
		];
		for (hour, y) in HOURS.into_iter().zip(approval) {
			assert_eq!(
				round(APP_ROOT.threshold(Perbill::from_rational(hour, 7u32 * 24u32))),
				y,
				"Approval mismatch at hour {hour}"
			);
		}
		for (hour, y) in HOURS.into_iter().zip(sup) {
			assert_eq!(
				round(SUP_ROOT.threshold(Perbill::from_rational(hour, 7u32 * 24u32))),
				y,
				"Support mismatch at hour {hour}"
			);
		}
	}
	#[test]
	fn whitelisted_caller_track() {
		const HOURS: [u32; 8] = [0, 1, 2, 3, 24, 36, 72, 84];
		let approval: [Perbill; 8] = [
			Perbill::from_rational(100, 100u32),
			Perbill::from_rational(994, 1000u32),
			Perbill::from_rational(9881, 10000u32),
			Perbill::from_rational(9821, 10000u32),
			Perbill::from_rational(8571, 10000u32),
			Perbill::from_rational(7857, 10000u32),
			Perbill::from_rational(5714, 10000u32),
			Perbill::from_rational(50, 100u32),
		];
		let sup: [Perbill; 8] = [
			Perbill::from_rational(50, 100u32),
			Perbill::from_rational(37, 1000u32),
			Perbill::from_rational(192, 10000u32),
			Perbill::from_rational(130, 10000u32),
			Perbill::from_rational(17, 10000u32),
			Perbill::from_rational(11, 10000u32),
			Perbill::from_rational(6, 10000u32),
			Perbill::from_rational(5, 10000u32),
		];
		for (hour, y) in HOURS.into_iter().zip(approval) {
			assert_eq!(
				round(APP_WHITELISTED.threshold(Perbill::from_rational(hour, 7u32 * 12u32))),
				y,
				"Approval mismatch at hour {hour}"
			);
		}
		for (hour, y) in HOURS.into_iter().zip(sup) {
			assert_eq!(
				round(SUP_WHITELISTED.threshold(Perbill::from_rational(hour, 7u32 * 12u32))),
				y,
				"Support mismatch at hour {hour}"
			);
		}
	}
	#[test]
	fn pool_admin_track() {
		const HOURS: [u32; 8] = [0, 1, 2, 3, 24, 36, 72, 84];
		let approval: [Perbill; 8] = [
			Perbill::from_rational(100, 100u32),
			Perbill::from_rational(9964, 10000u32),
			Perbill::from_rational(9929, 10000u32),
			Perbill::from_rational(9893, 10000u32),
			Perbill::from_rational(9143, 10000u32),
			Perbill::from_rational(8714, 10000u32),
			Perbill::from_rational(7429, 10000u32),
			Perbill::from_rational(70, 100u32),
		];
		let sup: [Perbill; 8] = [
			Perbill::from_rational(50, 100u32),
			Perbill::from_rational(1429, 10000u32),
			Perbill::from_rational(834, 10000u32),
			Perbill::from_rational(589, 10000u32),
			Perbill::from_rational(82, 10000u32),
			Perbill::from_rational(55, 10000u32),
			Perbill::from_rational(28, 10000u32),
			Perbill::from_rational(24, 10000u32),
		];
		for (hour, y) in HOURS.into_iter().zip(approval) {
			assert_eq!(
				round(APP_POOL_ADMIN.threshold(Perbill::from_rational(hour, 7u32 * 12u32))),
				y,
				"Approval mismatch at hour {hour}"
			);
		}
		for (hour, y) in HOURS.into_iter().zip(sup) {
			assert_eq!(
				round(SUP_POOL_ADMIN.threshold(Perbill::from_rational(hour, 7u32 * 12u32))),
				y,
				"Support mismatch at hour {hour}"
			);
		}
	}
	#[test]
	fn treasurer_track() {
		const HOURS: [u32; 8] = [0, 1, 2, 3, 24, 312, 324, 336];
		let approval: [Perbill; 8] = [
			Perbill::from_rational(100, 100u32),
			Perbill::from_rational(9991, 10000u32),
			Perbill::from_rational(9982, 10000u32),
			Perbill::from_rational(9973, 10000u32),
			Perbill::from_rational(9786, 10000u32),
			Perbill::from_rational(7214, 10000u32),
			Perbill::from_rational(7107, 10000u32),
			Perbill::from_rational(70, 100u32),
		];
		let sup: [Perbill; 8] = [
			Perbill::from_rational(50, 100u32),
			Perbill::from_rational(4984, 10000u32),
			Perbill::from_rational(4969, 10000u32),
			Perbill::from_rational(4953, 10000u32),
			Perbill::from_rational(4623, 10000u32),
			Perbill::from_rational(1, 100u32),
			Perbill::from_rational(1, 100u32),
			Perbill::from_rational(1, 100u32),
		];
		for (hour, y) in HOURS.into_iter().zip(approval) {
			assert_eq!(
				round(APP_TREASURER.threshold(Perbill::from_rational(hour, 14u32 * 24u32))),
				y,
				"Approval mismatch at hour {hour}"
			);
		}
		for (hour, y) in HOURS.into_iter().zip(sup) {
			assert_eq!(
				round(SUP_TREASURER.threshold(Perbill::from_rational(hour, 14u32 * 24u32))),
				y,
				"Support mismatch at hour {hour}"
			);
		}
	}
	#[test]
	fn ref_canceller_track() {
		const HOURS: [u32; 8] = [0, 1, 2, 3, 24, 36, 72, 84];
		let approval: [Perbill; 8] = [
			Perbill::from_rational(100, 100u32),
			Perbill::from_rational(9964, 10000u32),
			Perbill::from_rational(9929, 10000u32),
			Perbill::from_rational(9893, 10000u32),
			Perbill::from_rational(9143, 10000u32),
			Perbill::from_rational(8714, 10000u32),
			Perbill::from_rational(7429, 10000u32),
			Perbill::from_rational(70, 100u32),
		];
		let sup: [Perbill; 8] = [
			Perbill::from_rational(50, 100u32),
			Perbill::from_rational(1424, 10000u32),
			Perbill::from_rational(830, 10000u32),
			Perbill::from_rational(586, 10000u32),
			Perbill::from_rational(82, 10000u32),
			Perbill::from_rational(55, 10000u32),
			Perbill::from_rational(28, 10000u32),
			Perbill::from_rational(24, 10000u32),
		];
		for (hour, y) in HOURS.into_iter().zip(approval) {
			assert_eq!(
				round(APP_REF_CANCELLER.threshold(Perbill::from_rational(hour, 7u32 * 12u32))),
				y,
				"Approval mismatch at hour {hour}"
			);
		}
		for (hour, y) in HOURS.into_iter().zip(sup) {
			assert_eq!(
				round(SUP_REF_CANCELLER.threshold(Perbill::from_rational(hour, 7u32 * 12u32))),
				y,
				"Support mismatch at hour {hour}"
			);
		}
	}
	#[test]
	fn ref_killer_track() {
		const HOURS: [u32; 8] = [0, 1, 2, 3, 24, 36, 72, 84];
		let approval: [Perbill; 8] = [
			Perbill::from_rational(100, 100u32),
			Perbill::from_rational(9964, 10000u32),
			Perbill::from_rational(9929, 10000u32),
			Perbill::from_rational(9893, 10000u32),
			Perbill::from_rational(9143, 10000u32),
			Perbill::from_rational(8714, 10000u32),
			Perbill::from_rational(7429, 10000u32),
			Perbill::from_rational(70, 100u32),
		];
		let sup: [Perbill; 8] = [
			Perbill::from_rational(50, 100u32),
			Perbill::from_rational(1424, 10000u32),
			Perbill::from_rational(830, 10000u32),
			Perbill::from_rational(586, 10000u32),
			Perbill::from_rational(82, 10000u32),
			Perbill::from_rational(55, 10000u32),
			Perbill::from_rational(28, 10000u32),
			Perbill::from_rational(24, 10000u32),
		];
		for (hour, y) in HOURS.into_iter().zip(approval) {
			assert_eq!(
				round(APP_REF_KILLER.threshold(Perbill::from_rational(hour, 7u32 * 12u32))),
				y,
				"Approval mismatch at hour {hour}"
			);
		}
		for (hour, y) in HOURS.into_iter().zip(sup) {
			assert_eq!(
				round(SUP_REF_KILLER.threshold(Perbill::from_rational(hour, 7u32 * 12u32))),
				y,
				"Support mismatch at hour {hour}"
			);
		}
	}
}