auctions.rs 62.5 KB
Newer Older
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
// Copyright 2019-2020 Parity Technologies (UK) Ltd.
// This file is part of Polkadot.

// Polkadot is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.

// Polkadot is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
// GNU General Public License for more details.

// You should have received a copy of the GNU General Public License
// along with Polkadot.  If not, see <http://www.gnu.org/licenses/>.

//! Auctioning system to determine the set of Parachains in operation. This includes logic for the
//! auctioning mechanism and for reserving balance as part of the "payment". Unreserving the balance
//! happens elsewhere.

Shawn Tabrizi's avatar
Shawn Tabrizi committed
21
22
23
24
use crate::{
	slot_range::SlotRange,
	traits::{AuctionStatus, Auctioneer, LeaseError, Leaser, Registrar},
};
25
use frame_support::{
Shawn Tabrizi's avatar
Shawn Tabrizi committed
26
27
28
29
	dispatch::DispatchResult,
	ensure,
	traits::{Currency, Get, Randomness, ReservableCurrency},
	weights::Weight,
30
};
31
pub use pallet::*;
Shawn Tabrizi's avatar
Shawn Tabrizi committed
32
33
34
35
use parity_scale_codec::Decode;
use primitives::v1::Id as ParaId;
use sp_runtime::traits::{CheckedSub, One, Saturating, Zero};
use sp_std::{mem::swap, prelude::*};
36

37
38
39
type CurrencyOf<T> =
	<<T as Config>::Leaser as Leaser<<T as frame_system::Config>::BlockNumber>>::Currency;
type BalanceOf<T> = <<<T as Config>::Leaser as Leaser<<T as frame_system::Config>::BlockNumber>>::Currency as Currency<
Shawn Tabrizi's avatar
Shawn Tabrizi committed
40
41
	<T as frame_system::Config>::AccountId,
>>::Balance;
42
43
44
45

pub trait WeightInfo {
	fn new_auction() -> Weight;
	fn bid() -> Weight;
46
	fn cancel_auction() -> Weight;
47
48
49
50
51
	fn on_initialize() -> Weight;
}

pub struct TestWeightInfo;
impl WeightInfo for TestWeightInfo {
Shawn Tabrizi's avatar
Shawn Tabrizi committed
52
53
54
55
56
57
58
59
60
61
62
63
	fn new_auction() -> Weight {
		0
	}
	fn bid() -> Weight {
		0
	}
	fn cancel_auction() -> Weight {
		0
	}
	fn on_initialize() -> Weight {
		0
	}
64
65
66
67
68
}

/// An auction index. We count auctions in this type.
pub type AuctionIndex = u32;

69
70
71
type LeasePeriodOf<T> =
	<<T as Config>::Leaser as Leaser<<T as frame_system::Config>::BlockNumber>>::LeasePeriod;

72
// Winning data type. This encodes the top bidders of each range together with their bid.
Shawn Tabrizi's avatar
Shawn Tabrizi committed
73
74
type WinningData<T> = [Option<(<T as frame_system::Config>::AccountId, ParaId, BalanceOf<T>)>;
	SlotRange::SLOT_RANGE_COUNT];
75
76
// Winners data type. This encodes each of the final winners of a parachain auction, the parachain
// index assigned to them, their winning bid and the range that they won.
Shawn Tabrizi's avatar
Shawn Tabrizi committed
77
78
type WinnersData<T> =
	Vec<(<T as frame_system::Config>::AccountId, ParaId, BalanceOf<T>, SlotRange)>;
79

80
81
82
#[frame_support::pallet]
pub mod pallet {
	use super::*;
Shawn Tabrizi's avatar
Shawn Tabrizi committed
83
84
	use frame_support::{pallet_prelude::*, traits::EnsureOrigin, weights::DispatchClass};
	use frame_system::{ensure_root, ensure_signed, pallet_prelude::*};
85
86
87
88
89
90
91
92
93
94

	#[pallet::pallet]
	#[pallet::generate_store(pub(super) trait Store)]
	pub struct Pallet<T>(_);

	/// The module's configuration trait.
	#[pallet::config]
	pub trait Config: frame_system::Config {
		/// The overarching event type.
		type Event: From<Event<Self>> + IsType<<Self as frame_system::Config>::Event>;
95

96
		/// The type representing the leasing system.
97
98
99
100
101
		type Leaser: Leaser<
			Self::BlockNumber,
			AccountId = Self::AccountId,
			LeasePeriod = Self::BlockNumber,
		>;
102
103

		/// The parachain registrar type.
Shawn Tabrizi's avatar
Shawn Tabrizi committed
104
		type Registrar: Registrar<AccountId = Self::AccountId>;
105
106
107
108
109
110

		/// The number of blocks over which an auction may be retroactively ended.
		#[pallet::constant]
		type EndingPeriod: Get<Self::BlockNumber>;

		/// The length of each sample to take during the ending period.
111
		///
Denis_P's avatar
Denis_P committed
112
		/// `EndingPeriod` / `SampleLength` = Total # of Samples
113
114
115
116
117
118
119
120
121
122
123
		#[pallet::constant]
		type SampleLength: Get<Self::BlockNumber>;

		/// Something that provides randomness in the runtime.
		type Randomness: Randomness<Self::Hash, Self::BlockNumber>;

		/// The origin which may initiate auctions.
		type InitiateOrigin: EnsureOrigin<Self::Origin>;

		/// Weight Information for the Extrinsics in the Pallet
		type WeightInfo: WeightInfo;
124
125
	}

126
127
128
	#[pallet::event]
	#[pallet::generate_deposit(pub(super) fn deposit_event)]
	pub enum Event<T: Config> {
129
130
		/// An auction started. Provides its index and the block number where it will begin to
		/// close and the first lease period of the quadruplet that is auctioned.
Denis_P's avatar
Denis_P committed
131
		/// `[auction_index, lease_period, ending]`
132
		AuctionStarted(AuctionIndex, LeasePeriodOf<T>, T::BlockNumber),
Denis_P's avatar
Denis_P committed
133
		/// An auction ended. All funds become unreserved. `[auction_index]`
134
135
		AuctionClosed(AuctionIndex),
		/// Funds were reserved for a winning bid. First balance is the extra amount reserved.
Denis_P's avatar
Denis_P committed
136
		/// Second is the total. `[bidder, extra_reserved, total_amount]`
137
		Reserved(T::AccountId, BalanceOf<T>, BalanceOf<T>),
Denis_P's avatar
Denis_P committed
138
		/// Funds were unreserved since bidder is no longer active. `[bidder, amount]`
139
		Unreserved(T::AccountId, BalanceOf<T>),
140
141
		/// Someone attempted to lease the same slot twice for a parachain. The amount is held in reserve
		/// but no parachain slot has been leased.
Denis_P's avatar
Denis_P committed
142
		/// `[parachain_id, leaser, amount]`
143
		ReserveConfiscated(ParaId, T::AccountId, BalanceOf<T>),
144
		/// A new bid has been accepted as the current winner.
Denis_P's avatar
Denis_P committed
145
		/// `[who, para_id, amount, first_slot, last_slot]`
146
		BidAccepted(T::AccountId, ParaId, BalanceOf<T>, LeasePeriodOf<T>, LeasePeriodOf<T>),
147
		/// The winning offset was chosen for an auction. This will map into the `Winning` storage map.
Denis_P's avatar
Denis_P committed
148
		/// `[auction_index, block_number]`
149
		WinningOffset(AuctionIndex, T::BlockNumber),
150
151
	}

152
153
	#[pallet::error]
	pub enum Error<T> {
154
155
156
157
		/// This auction is already in progress.
		AuctionInProgress,
		/// The lease period is in the past.
		LeasePeriodInPast,
158
159
		/// Para is not registered
		ParaNotRegistered,
160
161
162
163
164
165
		/// Not a current auction.
		NotCurrentAuction,
		/// Not an auction.
		NotAuction,
		/// Auction has already ended.
		AuctionEnded,
166
167
		/// The para is already leased out for part of this range.
		AlreadyLeasedOut,
168
169
	}

170
171
172
173
	/// Number of auctions started so far.
	#[pallet::storage]
	#[pallet::getter(fn auction_counter)]
	pub type AuctionCounter<T> = StorageValue<_, AuctionIndex, ValueQuery>;
174

175
176
177
178
179
180
181
182
183
184
185
186
187
	/// Information relating to the current auction, if there is one.
	///
	/// The first item in the tuple is the lease period index that the first of the four
	/// contiguous lease periods on auction is for. The second is the block number when the
	/// auction will "begin to end", i.e. the first block of the Ending Period of the auction.
	#[pallet::storage]
	#[pallet::getter(fn auction_info)]
	pub type AuctionInfo<T: Config> = StorageValue<_, (LeasePeriodOf<T>, T::BlockNumber)>;

	/// Amounts currently reserved in the accounts of the bidders currently winning
	/// (sub-)ranges.
	#[pallet::storage]
	#[pallet::getter(fn reserved_amounts)]
Shawn Tabrizi's avatar
Shawn Tabrizi committed
188
189
	pub type ReservedAmounts<T: Config> =
		StorageMap<_, Twox64Concat, (T::AccountId, ParaId), BalanceOf<T>>;
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204

	/// The winning bids for each of the 10 ranges at each sample in the final Ending Period of
	/// the current auction. The map's key is the 0-based index into the Sample Size. The
	/// first sample of the ending period is 0; the last is `Sample Size - 1`.
	#[pallet::storage]
	#[pallet::getter(fn winning)]
	pub type Winning<T: Config> = StorageMap<_, Twox64Concat, T::BlockNumber, WinningData<T>>;

	#[pallet::extra_constants]
	impl<T: Config> Pallet<T> {
		//TODO: rename to snake case after https://github.com/paritytech/substrate/issues/8826 fixed.
		#[allow(non_snake_case)]
		fn SlotRangeCount() -> u32 {
			SlotRange::SLOT_RANGE_COUNT as u32
		}
205

206
207
208
209
210
211
		//TODO: rename to snake case after https://github.com/paritytech/substrate/issues/8826 fixed.
		#[allow(non_snake_case)]
		fn LeasePeriodsPerSlot() -> u32 {
			SlotRange::LEASE_PERIODS_PER_SLOT as u32
		}
	}
212

213
214
	#[pallet::hooks]
	impl<T: Config> Hooks<BlockNumberFor<T>> for Pallet<T> {
215
216
217
218
219
220
		fn on_initialize(n: T::BlockNumber) -> Weight {
			let mut weight = T::DbWeight::get().reads(1);

			// If the current auction was in its ending period last block, then ensure that the (sub-)range
			// winner information is duplicated from the previous block in case no bids happened in the
			// last block.
221
			if let AuctionStatus::EndingPeriod(offset, _sub_sample) = Self::auction_status(n) {
222
223
224
				weight = weight.saturating_add(T::DbWeight::get().reads(1));
				if !Winning::<T>::contains_key(&offset) {
					weight = weight.saturating_add(T::DbWeight::get().writes(1));
Shawn Tabrizi's avatar
Shawn Tabrizi committed
225
226
227
228
					let winning_data = offset
						.checked_sub(&One::one())
						.and_then(Winning::<T>::get)
						.unwrap_or([Self::EMPTY; SlotRange::SLOT_RANGE_COUNT]);
229
230
231
232
233
234
235
236
					Winning::<T>::insert(offset, winning_data);
				}
			}

			// Check to see if an auction just ended.
			if let Some((winning_ranges, auction_lease_period_index)) = Self::check_auction_end(n) {
				// Auction is ended now. We have the winning ranges and the lease period index which
				// acts as the offset. Handle it.
Shawn Tabrizi's avatar
Shawn Tabrizi committed
237
				Self::manage_auction_end(auction_lease_period_index, winning_ranges);
238
239
240
241
242
				weight = weight.saturating_add(T::WeightInfo::on_initialize());
			}

			weight
		}
243
	}
244

245
246
	#[pallet::call]
	impl<T: Config> Pallet<T> {
247
248
249
250
251
		/// Create a new auction.
		///
		/// This can only happen when there isn't already an auction in progress and may only be
		/// called by the root origin. Accepts the `duration` of this auction and the
		/// `lease_period_index` of the initial lease period of the four that are to be auctioned.
252
253
254
255
256
257
		#[pallet::weight((T::WeightInfo::new_auction(), DispatchClass::Operational))]
		pub fn new_auction(
			origin: OriginFor<T>,
			#[pallet::compact] duration: T::BlockNumber,
			#[pallet::compact] lease_period_index: LeasePeriodOf<T>,
		) -> DispatchResult {
258
			T::InitiateOrigin::ensure_origin(origin)?;
259
			Self::do_new_auction(duration, lease_period_index)
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
		}

		/// Make a new bid from an account (including a parachain account) for deploying a new
		/// parachain.
		///
		/// Multiple simultaneous bids from the same bidder are allowed only as long as all active
		/// bids overlap each other (i.e. are mutually exclusive). Bids cannot be redacted.
		///
		/// - `sub` is the sub-bidder ID, allowing for multiple competing bids to be made by (and
		/// funded by) the same account.
		/// - `auction_index` is the index of the auction to bid on. Should just be the present
		/// value of `AuctionCounter`.
		/// - `first_slot` is the first lease period index of the range to bid on. This is the
		/// absolute lease period index value, not an auction-specific offset.
		/// - `last_slot` is the last lease period index of the range to bid on. This is the
		/// absolute lease period index value, not an auction-specific offset.
		/// - `amount` is the amount to bid to be held as deposit for the parachain should the
		/// bid win. This amount is held throughout the range.
278
279
280
281
282
283
284
		#[pallet::weight(T::WeightInfo::bid())]
		pub fn bid(
			origin: OriginFor<T>,
			#[pallet::compact] para: ParaId,
			#[pallet::compact] auction_index: AuctionIndex,
			#[pallet::compact] first_slot: LeasePeriodOf<T>,
			#[pallet::compact] last_slot: LeasePeriodOf<T>,
Shawn Tabrizi's avatar
Shawn Tabrizi committed
285
			#[pallet::compact] amount: BalanceOf<T>,
286
		) -> DispatchResult {
287
288
			let who = ensure_signed(origin)?;
			Self::handle_bid(who, para, auction_index, first_slot, last_slot, amount)?;
289
			Ok(())
290
		}
291
292
293
294

		/// Cancel an in-progress auction.
		///
		/// Can only be called by Root origin.
295
296
		#[pallet::weight(T::WeightInfo::cancel_auction())]
		pub fn cancel_auction(origin: OriginFor<T>) -> DispatchResult {
297
298
299
300
301
			ensure_root(origin)?;
			// Unreserve all bids.
			for ((bidder, _), amount) in ReservedAmounts::<T>::drain() {
				CurrencyOf::<T>::unreserve(&bidder, amount);
			}
302
			Winning::<T>::remove_all(None);
303
			AuctionInfo::<T>::kill();
304
			Ok(())
305
		}
306
307
308
	}
}

309
impl<T: Config> Auctioneer<T::BlockNumber> for Pallet<T> {
310
311
312
313
314
315
316
317
318
319
320
	type AccountId = T::AccountId;
	type LeasePeriod = T::BlockNumber;
	type Currency = CurrencyOf<T>;

	fn new_auction(
		duration: T::BlockNumber,
		lease_period_index: LeasePeriodOf<T>,
	) -> DispatchResult {
		Self::do_new_auction(duration, lease_period_index)
	}

321
	// Returns the status of the auction given the current block number.
322
	fn auction_status(now: T::BlockNumber) -> AuctionStatus<T::BlockNumber> {
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
		let early_end = match AuctionInfo::<T>::get() {
			Some((_, early_end)) => early_end,
			None => return AuctionStatus::NotStarted,
		};

		let after_early_end = match now.checked_sub(&early_end) {
			Some(after_early_end) => after_early_end,
			None => return AuctionStatus::StartingPeriod,
		};

		let ending_period = T::EndingPeriod::get();
		if after_early_end < ending_period {
			let sample_length = T::SampleLength::get().max(One::one());
			let sample = after_early_end / sample_length;
			let sub_sample = after_early_end % sample_length;
			return AuctionStatus::EndingPeriod(sample, sub_sample)
		} else {
			// This is safe because of the comparison operator above
			return AuctionStatus::VrfDelay(after_early_end - ending_period)
342
343
344
345
346
347
348
349
350
351
		}
	}

	fn place_bid(
		bidder: T::AccountId,
		para: ParaId,
		first_slot: LeasePeriodOf<T>,
		last_slot: LeasePeriodOf<T>,
		amount: BalanceOf<T>,
	) -> DispatchResult {
352
		Self::handle_bid(bidder, para, AuctionCounter::<T>::get(), first_slot, last_slot, amount)
353
354
	}

355
356
	fn lease_period_index(b: T::BlockNumber) -> Option<(Self::LeasePeriod, bool)> {
		T::Leaser::lease_period_index(b)
357
	}
358

359
360
361
	#[cfg(any(feature = "runtime-benchmarks", test))]
	fn lease_period_length() -> (T::BlockNumber, T::BlockNumber) {
		T::Leaser::lease_period_length()
362
363
364
365
366
	}

	fn has_won_an_auction(para: ParaId, bidder: &T::AccountId) -> bool {
		!T::Leaser::deposit_held(para, bidder).is_zero()
	}
367
368
}

369
impl<T: Config> Pallet<T> {
370
371
372
	// A trick to allow me to initialize large arrays with nothing in them.
	const EMPTY: Option<(<T as frame_system::Config>::AccountId, ParaId, BalanceOf<T>)> = None;

373
374
375
376
377
378
379
380
381
	/// Create a new auction.
	///
	/// This can only happen when there isn't already an auction in progress. Accepts the `duration`
	/// of this auction and the `lease_period_index` of the initial lease period of the four that
	/// are to be auctioned.
	fn do_new_auction(
		duration: T::BlockNumber,
		lease_period_index: LeasePeriodOf<T>,
	) -> DispatchResult {
382
383
		let maybe_auction = AuctionInfo::<T>::get();
		ensure!(maybe_auction.is_none(), Error::<T>::AuctionInProgress);
384
385
386
387
388
		let now = frame_system::Pallet::<T>::block_number();
		if let Some((current_lease_period, _)) = T::Leaser::lease_period_index(now) {
			// If there is no active lease period, then we don't need to make this check.
			ensure!(lease_period_index >= current_lease_period, Error::<T>::LeasePeriodInPast);
		}
389
390

		// Bump the counter.
Shawn Tabrizi's avatar
Shawn Tabrizi committed
391
392
393
394
		let n = AuctionCounter::<T>::mutate(|n| {
			*n += 1;
			*n
		});
395
396

		// Set the information.
397
		let ending = frame_system::Pallet::<T>::block_number().saturating_add(duration);
398
399
		AuctionInfo::<T>::put((lease_period_index, ending));

400
		Self::deposit_event(Event::<T>::AuctionStarted(n, lease_period_index, ending));
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
		Ok(())
	}

	/// Actually place a bid in the current auction.
	///
	/// - `bidder`: The account that will be funding this bid.
	/// - `auction_index`: The auction index of the bid. For this to succeed, must equal
	/// the current value of `AuctionCounter`.
	/// - `first_slot`: The first lease period index of the range to be bid on.
	/// - `last_slot`: The last lease period index of the range to be bid on (inclusive).
	/// - `amount`: The total amount to be the bid for deposit over the range.
	pub fn handle_bid(
		bidder: T::AccountId,
		para: ParaId,
		auction_index: u32,
		first_slot: LeasePeriodOf<T>,
		last_slot: LeasePeriodOf<T>,
		amount: BalanceOf<T>,
	) -> DispatchResult {
420
421
		// Ensure para is registered before placing a bid on it.
		ensure!(T::Registrar::is_registered(para), Error::<T>::ParaNotRegistered);
422
		// Bidding on latest auction.
423
		ensure!(auction_index == AuctionCounter::<T>::get(), Error::<T>::NotCurrentAuction);
424
		// Assume it's actually an auction (this should never fail because of above).
425
		let (first_lease_period, _) = AuctionInfo::<T>::get().ok_or(Error::<T>::NotAuction)?;
426

427
428
429
430
431
432
433
434
435
436
		// Get the auction status and the current sample block. For the starting period, the sample
		// block is zero.
		let auction_status = Self::auction_status(frame_system::Pallet::<T>::block_number());
		// The offset into the ending samples of the auction.
		let offset = match auction_status {
			AuctionStatus::NotStarted => return Err(Error::<T>::AuctionEnded.into()),
			AuctionStatus::StartingPeriod => Zero::zero(),
			AuctionStatus::EndingPeriod(o, _) => o,
			AuctionStatus::VrfDelay(_) => return Err(Error::<T>::AuctionEnded.into()),
		};
437

438
		// We also make sure that the bid is not for any existing leases the para already has.
Shawn Tabrizi's avatar
Shawn Tabrizi committed
439
440
441
442
		ensure!(
			!T::Leaser::already_leased(para, first_slot, last_slot),
			Error::<T>::AlreadyLeasedOut
		);
443

444
445
446
447
		// Our range.
		let range = SlotRange::new_bounded(first_lease_period, first_slot, last_slot)?;
		// Range as an array index.
		let range_index = range as u8 as usize;
448

449
450
451
		// The current winning ranges.
		let mut current_winning = Winning::<T>::get(offset)
			.or_else(|| offset.checked_sub(&One::one()).and_then(Winning::<T>::get))
452
			.unwrap_or([Self::EMPTY; SlotRange::SLOT_RANGE_COUNT]);
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474

		// If this bid beat the previous winner of our range.
		if current_winning[range_index].as_ref().map_or(true, |last| amount > last.2) {
			// Ok; we are the new winner of this range - reserve the additional amount and record.

			// Get the amount already held on deposit if this is a renewal bid (i.e. there's
			// an existing lease on the same para by the same leaser).
			let existing_lease_deposit = T::Leaser::deposit_held(para, &bidder);
			let reserve_required = amount.saturating_sub(existing_lease_deposit);

			// Get the amount already reserved in any prior and still active bids by us.
			let bidder_para = (bidder.clone(), para);
			let already_reserved = ReservedAmounts::<T>::get(&bidder_para).unwrap_or_default();

			// If these don't already cover the bid...
			if let Some(additional) = reserve_required.checked_sub(&already_reserved) {
				// ...then reserve some more funds from their account, failing if there's not
				// enough funds.
				CurrencyOf::<T>::reserve(&bidder, additional)?;
				// ...and record the amount reserved.
				ReservedAmounts::<T>::insert(&bidder_para, reserve_required);

475
				Self::deposit_event(Event::<T>::Reserved(
476
477
478
479
480
481
					bidder.clone(),
					additional,
					reserve_required,
				));
			}

482
483
			// Return any funds reserved for the previous winner if we are not in the ending period
			// and they no longer have any active bids.
484
485
486
			let mut outgoing_winner = Some((bidder.clone(), para, amount));
			swap(&mut current_winning[range_index], &mut outgoing_winner);
			if let Some((who, para, _amount)) = outgoing_winner {
Shawn Tabrizi's avatar
Shawn Tabrizi committed
487
488
489
490
491
				if auction_status.is_starting() &&
					current_winning
						.iter()
						.filter_map(Option::as_ref)
						.all(|&(ref other, other_para, _)| other != &who || other_para != para)
492
493
494
495
496
497
				{
					// Previous bidder is no longer winning any ranges: unreserve their funds.
					if let Some(amount) = ReservedAmounts::<T>::take(&(who.clone(), para)) {
						// It really should be reserved; there's not much we can do here on fail.
						let err_amt = CurrencyOf::<T>::unreserve(&who, amount);
						debug_assert!(err_amt.is_zero());
498
						Self::deposit_event(Event::<T>::Unreserved(who, amount));
499
500
501
502
503
504
					}
				}
			}

			// Update the range winner.
			Winning::<T>::insert(offset, &current_winning);
Shawn Tabrizi's avatar
Shawn Tabrizi committed
505
506
507
			Self::deposit_event(Event::<T>::BidAccepted(
				bidder, para, amount, first_slot, last_slot,
			));
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
		}
		Ok(())
	}

	/// Some when the auction's end is known (with the end block number). None if it is unknown.
	/// If `Some` then the block number must be at most the previous block and at least the
	/// previous block minus `T::EndingPeriod::get()`.
	///
	/// This mutates the state, cleaning up `AuctionInfo` and `Winning` in the case of an auction
	/// ending. An immediately subsequent call with the same argument will always return `None`.
	fn check_auction_end(now: T::BlockNumber) -> Option<(WinningData<T>, LeasePeriodOf<T>)> {
		if let Some((lease_period_index, early_end)) = AuctionInfo::<T>::get() {
			let ending_period = T::EndingPeriod::get();
			let late_end = early_end.saturating_add(ending_period);
			let is_ended = now >= late_end;
			if is_ended {
				// auction definitely ended.
				// check to see if we can determine the actual ending point.
				let (raw_offset, known_since) = T::Randomness::random(&b"para_auction"[..]);

				if late_end <= known_since {
					// Our random seed was known only after the auction ended. Good to use.
Shawn Tabrizi's avatar
Shawn Tabrizi committed
530
531
532
533
534
535
					let raw_offset_block_number = <T::BlockNumber>::decode(
						&mut raw_offset.as_ref(),
					)
					.expect("secure hashes should always be bigger than the block number; qed");
					let offset = (raw_offset_block_number % ending_period) /
						T::SampleLength::get().max(One::one());
536

537
538
					let auction_counter = AuctionCounter::<T>::get();
					Self::deposit_event(Event::<T>::WinningOffset(auction_counter, offset));
Shawn Tabrizi's avatar
Shawn Tabrizi committed
539
540
					let res = Winning::<T>::get(offset)
						.unwrap_or([Self::EMPTY; SlotRange::SLOT_RANGE_COUNT]);
541
542
					// This `remove_all` statement should remove at most `EndingPeriod` / `SampleLength` items,
					// which should be bounded and sensibly configured in the runtime.
543
					Winning::<T>::remove_all(None);
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
					AuctionInfo::<T>::kill();
					return Some((res, lease_period_index))
				}
			}
		}
		None
	}

	/// Auction just ended. We have the current lease period, the auction's lease period (which
	/// is guaranteed to be at least the current period) and the bidders that were winning each
	/// range at the time of the auction's close.
	fn manage_auction_end(
		auction_lease_period_index: LeasePeriodOf<T>,
		winning_ranges: WinningData<T>,
	) {
		// First, unreserve all amounts that were reserved for the bids. We will later re-reserve the
		// amounts from the bidders that ended up being assigned the slot so there's no need to
		// special-case them here.
562
		for ((bidder, _), amount) in ReservedAmounts::<T>::drain() {
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
			CurrencyOf::<T>::unreserve(&bidder, amount);
		}

		// Next, calculate the winning combination of slots and thus the final winners of the
		// auction.
		let winners = Self::calculate_winners(winning_ranges);

		// Go through those winners and re-reserve their bid, updating our table of deposits
		// accordingly.
		for (leaser, para, amount, range) in winners.into_iter() {
			let begin_offset = LeasePeriodOf::<T>::from(range.as_pair().0 as u32);
			let period_begin = auction_lease_period_index + begin_offset;
			let period_count = LeasePeriodOf::<T>::from(range.len() as u32);

			match T::Leaser::lease_out(para, &leaser, amount, period_begin, period_count) {
578
579
580
				Err(LeaseError::ReserveFailed) |
				Err(LeaseError::AlreadyEnded) |
				Err(LeaseError::NoLeasePeriod) => {
581
582
					// Should never happen since we just unreserved this amount (and our offset is from the
					// present period). But if it does, there's not much we can do.
Shawn Tabrizi's avatar
Shawn Tabrizi committed
583
				},
584
585
586
587
				Err(LeaseError::AlreadyLeased) => {
					// The leaser attempted to get a second lease on the same para ID, possibly griefing us. Let's
					// keep the amount reserved and let governance sort it out.
					if CurrencyOf::<T>::reserve(&leaser, amount).is_ok() {
588
						Self::deposit_event(Event::<T>::ReserveConfiscated(para, leaser, amount));
589
					}
Shawn Tabrizi's avatar
Shawn Tabrizi committed
590
				},
591
592
593
594
				Ok(()) => {}, // Nothing to report.
			}
		}

595
		Self::deposit_event(Event::<T>::AuctionClosed(AuctionCounter::<T>::get()));
596
597
598
599
600
	}

	/// Calculate the final winners from the winning slots.
	///
	/// This is a simple dynamic programming algorithm designed by Al, the original code is at:
Denis_P's avatar
Denis_P committed
601
	/// `https://github.com/w3f/consensus/blob/master/NPoS/auctiondynamicthing.py`
Shawn Tabrizi's avatar
Shawn Tabrizi committed
602
	fn calculate_winners(mut winning: WinningData<T>) -> WinnersData<T> {
603
		let winning_ranges = {
Shawn Tabrizi's avatar
Shawn Tabrizi committed
604
605
			let mut best_winners_ending_at: [(Vec<SlotRange>, BalanceOf<T>);
				SlotRange::LEASE_PERIODS_PER_SLOT] = Default::default();
606
			let best_bid = |range: SlotRange| {
Shawn Tabrizi's avatar
Shawn Tabrizi committed
607
608
				winning[range as u8 as usize]
					.as_ref()
609
610
					.map(|(_, _, amount)| *amount * (range.len() as u32).into())
			};
611
			for i in 0..SlotRange::LEASE_PERIODS_PER_SLOT {
612
				let r = SlotRange::new_bounded(0, 0, i as u32).expect("`i < LPPS`; qed");
613
614
615
616
617
				if let Some(bid) = best_bid(r) {
					best_winners_ending_at[i] = (vec![r], bid);
				}
				for j in 0..i {
					let r = SlotRange::new_bounded(0, j as u32 + 1, i as u32)
618
						.expect("`i < LPPS`; `j < i`; `j + 1 < LPPS`; qed");
619
620
621
622
623
624
625
626
627
628
629
630
631
632
					if let Some(mut bid) = best_bid(r) {
						bid += best_winners_ending_at[j].1;
						if bid > best_winners_ending_at[i].1 {
							let mut new_winners = best_winners_ending_at[j].0.clone();
							new_winners.push(r);
							best_winners_ending_at[i] = (new_winners, bid);
						}
					} else {
						if best_winners_ending_at[j].1 > best_winners_ending_at[i].1 {
							best_winners_ending_at[i] = best_winners_ending_at[j].clone();
						}
					}
				}
			}
633
			best_winners_ending_at[SlotRange::LEASE_PERIODS_PER_SLOT - 1].0.clone()
634
635
		};

Shawn Tabrizi's avatar
Shawn Tabrizi committed
636
637
638
639
640
641
642
643
644
645
646
647
648
649
		winning_ranges
			.into_iter()
			.map(|range| {
				let mut final_winner = Default::default();
				swap(
					&mut final_winner,
					winning[range as u8 as usize]
						.as_mut()
						.expect("none values are filtered out in previous logic; qed"),
				);
				let (bidder, para, amount) = final_winner;
				(bidder, para, amount, range)
			})
			.collect::<Vec<_>>()
650
651
652
653
654
655
656
	}
}

/// tests for this module
#[cfg(test)]
mod tests {
	use super::*;
Shawn Tabrizi's avatar
Shawn Tabrizi committed
657
	use crate::{auctions, mock::TestRegistrar};
658
	use frame_support::{
Shawn Tabrizi's avatar
Shawn Tabrizi committed
659
		assert_noop, assert_ok, assert_storage_noop,
660
		dispatch::DispatchError::BadOrigin,
Shawn Tabrizi's avatar
Shawn Tabrizi committed
661
662
		ord_parameter_types, parameter_types,
		traits::{OnFinalize, OnInitialize},
663
	};
Shawn Tabrizi's avatar
Shawn Tabrizi committed
664
	use frame_system::{EnsureOneOf, EnsureRoot, EnsureSignedBy};
665
666
	use pallet_balances;
	use primitives::v1::{BlockNumber, Header, Id as ParaId};
Shawn Tabrizi's avatar
Shawn Tabrizi committed
667
668
669
	use sp_core::H256;
	use sp_runtime::traits::{BlakeTwo256, IdentityLookup};
	use std::{cell::RefCell, collections::BTreeMap};
670
671
672
673
674
675
676
677
678
679

	type UncheckedExtrinsic = frame_system::mocking::MockUncheckedExtrinsic<Test>;
	type Block = frame_system::mocking::MockBlock<Test>;

	frame_support::construct_runtime!(
		pub enum Test where
			Block = Block,
			NodeBlock = Block,
			UncheckedExtrinsic = UncheckedExtrinsic,
		{
680
681
682
			System: frame_system::{Pallet, Call, Config, Storage, Event<T>},
			Balances: pallet_balances::{Pallet, Call, Storage, Config<T>, Event<T>},
			Auctions: auctions::{Pallet, Call, Storage, Event<T>},
683
684
685
686
687
688
689
		}
	);

	parameter_types! {
		pub const BlockHashCount: u32 = 250;
	}
	impl frame_system::Config for Test {
690
		type BaseCallFilter = frame_support::traits::Everything;
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
		type BlockWeights = ();
		type BlockLength = ();
		type DbWeight = ();
		type Origin = Origin;
		type Call = Call;
		type Index = u64;
		type BlockNumber = BlockNumber;
		type Hash = H256;
		type Hashing = BlakeTwo256;
		type AccountId = u64;
		type Lookup = IdentityLookup<Self::AccountId>;
		type Header = Header;
		type Event = Event;
		type BlockHashCount = BlockHashCount;
		type Version = ();
		type PalletInfo = PalletInfo;
		type AccountData = pallet_balances::AccountData<u64>;
		type OnNewAccount = ();
		type OnKilledAccount = ();
		type SystemWeightInfo = ();
		type SS58Prefix = ();
712
		type OnSetCode = ();
713
714
715
716
	}

	parameter_types! {
		pub const ExistentialDeposit: u64 = 1;
Gavin Wood's avatar
Gavin Wood committed
717
		pub const MaxReserves: u32 = 50;
718
719
720
721
722
723
724
725
726
727
	}

	impl pallet_balances::Config for Test {
		type Balance = u64;
		type DustRemoval = ();
		type Event = Event;
		type ExistentialDeposit = ExistentialDeposit;
		type AccountStore = System;
		type WeightInfo = ();
		type MaxLocks = ();
Gavin Wood's avatar
Gavin Wood committed
728
729
		type MaxReserves = MaxReserves;
		type ReserveIdentifier = [u8; 8];
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
	}

	#[derive(Eq, PartialEq, Ord, PartialOrd, Clone, Copy, Debug)]
	pub struct LeaseData {
		leaser: u64,
		amount: u64,
	}

	thread_local! {
		pub static LEASES:
			RefCell<BTreeMap<(ParaId, BlockNumber), LeaseData>> = RefCell::new(BTreeMap::new());
	}

	fn leases() -> Vec<((ParaId, BlockNumber), LeaseData)> {
		LEASES.with(|p| (&*p.borrow()).clone().into_iter().collect::<Vec<_>>())
	}

	pub struct TestLeaser;
748
	impl Leaser<BlockNumber> for TestLeaser {
749
750
751
752
753
754
755
756
757
758
759
760
761
		type AccountId = u64;
		type LeasePeriod = BlockNumber;
		type Currency = Balances;

		fn lease_out(
			para: ParaId,
			leaser: &Self::AccountId,
			amount: <Self::Currency as Currency<Self::AccountId>>::Balance,
			period_begin: Self::LeasePeriod,
			period_count: Self::LeasePeriod,
		) -> Result<(), LeaseError> {
			LEASES.with(|l| {
				let mut leases = l.borrow_mut();
762
763
764
765
				let now = System::block_number();
				let (current_lease_period, _) =
					Self::lease_period_index(now).ok_or(LeaseError::NoLeasePeriod)?;
				if period_begin < current_lease_period {
766
767
768
769
770
771
772
773
774
775
776
777
					return Err(LeaseError::AlreadyEnded)
				}
				for period in period_begin..(period_begin + period_count) {
					if leases.contains_key(&(para, period)) {
						return Err(LeaseError::AlreadyLeased)
					}
					leases.insert((para, period), LeaseData { leaser: leaser.clone(), amount });
				}
				Ok(())
			})
		}

778
779
		fn deposit_held(
			para: ParaId,
Shawn Tabrizi's avatar
Shawn Tabrizi committed
780
			leaser: &Self::AccountId,
781
		) -> <Self::Currency as Currency<Self::AccountId>>::Balance {
Shawn Tabrizi's avatar
Shawn Tabrizi committed
782
783
784
785
786
787
788
789
790
			leases()
				.iter()
				.filter_map(|((id, _period), data)| {
					if id == &para && &data.leaser == leaser {
						Some(data.amount)
					} else {
						None
					}
				})
791
792
793
794
				.max()
				.unwrap_or_default()
		}

795
796
		fn lease_period_length() -> (BlockNumber, BlockNumber) {
			(10, 0)
797
798
		}

799
800
801
802
803
804
805
806
		fn lease_period_index(b: BlockNumber) -> Option<(Self::LeasePeriod, bool)> {
			let (lease_period_length, offset) = Self::lease_period_length();
			let b = b.checked_sub(offset)?;

			let lease_period = b / lease_period_length;
			let first_block = (b % lease_period_length).is_zero();

			Some((lease_period, first_block))
807
		}
808
809
810
811

		fn already_leased(
			para_id: ParaId,
			first_period: Self::LeasePeriod,
Shawn Tabrizi's avatar
Shawn Tabrizi committed
812
			last_period: Self::LeasePeriod,
813
814
		) -> bool {
			leases().into_iter().any(|((para, period), _data)| {
Shawn Tabrizi's avatar
Shawn Tabrizi committed
815
				para == para_id && first_period <= period && period <= last_period
816
817
			})
		}
818
819
	}

Shawn Tabrizi's avatar
Shawn Tabrizi committed
820
	ord_parameter_types! {
821
822
823
		pub const Six: u64 = 6;
	}

Shawn Tabrizi's avatar
Shawn Tabrizi committed
824
	type RootOrSix = EnsureOneOf<u64, EnsureRoot<u64>, EnsureSignedBy<Six, u64>>;
825
826
827
828
829
830
831
832
833
834
835
836
837
838

	thread_local! {
		pub static LAST_RANDOM: RefCell<Option<(H256, u32)>> = RefCell::new(None);
	}
	fn set_last_random(output: H256, known_since: u32) {
		LAST_RANDOM.with(|p| *p.borrow_mut() = Some((output, known_since)))
	}
	pub struct TestPastRandomness;
	impl Randomness<H256, BlockNumber> for TestPastRandomness {
		fn random(_subject: &[u8]) -> (H256, u32) {
			LAST_RANDOM.with(|p| {
				if let Some((output, known_since)) = &*p.borrow() {
					(*output, *known_since)
				} else {
839
					(H256::zero(), frame_system::Pallet::<Test>::block_number())
840
841
842
843
844
				}
			})
		}
	}

Shawn Tabrizi's avatar
Shawn Tabrizi committed
845
	parameter_types! {
846
847
848
849
		pub static EndingPeriod: BlockNumber = 3;
		pub static SampleLength: BlockNumber = 1;
	}

850
851
852
	impl Config for Test {
		type Event = Event;
		type Leaser = TestLeaser;
853
		type Registrar = TestRegistrar<Self>;
854
		type EndingPeriod = EndingPeriod;
855
		type SampleLength = SampleLength;
856
857
858
859
860
861
862
863
864
		type Randomness = TestPastRandomness;
		type InitiateOrigin = RootOrSix;
		type WeightInfo = crate::auctions::TestWeightInfo;
	}

	// This function basically just builds a genesis storage key/value store according to
	// our desired mock up.
	pub fn new_test_ext() -> sp_io::TestExternalities {
		let mut t = frame_system::GenesisConfig::default().build_storage::<Test>().unwrap();
Shawn Tabrizi's avatar
Shawn Tabrizi committed
865
		pallet_balances::GenesisConfig::<Test> {
866
			balances: vec![(1, 10), (2, 20), (3, 30), (4, 40), (5, 50), (6, 60)],
Shawn Tabrizi's avatar
Shawn Tabrizi committed
867
868
869
		}
		.assimilate_storage(&mut t)
		.unwrap();
870
871
872
		let mut ext: sp_io::TestExternalities = t.into();
		ext.execute_with(|| {
			// Register para 0, 1, 2, and 3 for tests
Shawn Tabrizi's avatar
Shawn Tabrizi committed
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
			assert_ok!(TestRegistrar::<Test>::register(
				1,
				0.into(),
				Default::default(),
				Default::default()
			));
			assert_ok!(TestRegistrar::<Test>::register(
				1,
				1.into(),
				Default::default(),
				Default::default()
			));
			assert_ok!(TestRegistrar::<Test>::register(
				1,
				2.into(),
				Default::default(),
				Default::default()
			));
			assert_ok!(TestRegistrar::<Test>::register(
				1,
				3.into(),
				Default::default(),
				Default::default()
			));
897
898
		});
		ext
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
	}

	fn run_to_block(n: BlockNumber) {
		while System::block_number() < n {
			Auctions::on_finalize(System::block_number());
			Balances::on_finalize(System::block_number());
			System::on_finalize(System::block_number());
			System::set_block_number(System::block_number() + 1);
			System::on_initialize(System::block_number());
			Balances::on_initialize(System::block_number());
			Auctions::on_initialize(System::block_number());
		}
	}

	#[test]
	fn basic_setup_works() {
		new_test_ext().execute_with(|| {
916
			assert_eq!(AuctionCounter::<Test>::get(), 0);
917
			assert_eq!(TestLeaser::deposit_held(0u32.into(), &1), 0);
Shawn Tabrizi's avatar
Shawn Tabrizi committed
918
919
920
921
			assert_eq!(
				Auctions::auction_status(System::block_number()),
				AuctionStatus::<u32>::NotStarted
			);
922
923
924

			run_to_block(10);

925
			assert_eq!(AuctionCounter::<Test>::get(), 0);
926
			assert_eq!(TestLeaser::deposit_held(0u32.into(), &1), 0);
Shawn Tabrizi's avatar
Shawn Tabrizi committed
927
928
929
930
			assert_eq!(
				Auctions::auction_status(System::block_number()),
				AuctionStatus::<u32>::NotStarted
			);
931
932
933
934
935
936
937
938
939
940
941
		});
	}

	#[test]
	fn can_start_auction() {
		new_test_ext().execute_with(|| {
			run_to_block(1);

			assert_noop!(Auctions::new_auction(Origin::signed(1), 5, 1), BadOrigin);
			assert_ok!(Auctions::new_auction(Origin::signed(6), 5, 1));

942
			assert_eq!(AuctionCounter::<Test>::get(), 1);
Shawn Tabrizi's avatar
Shawn Tabrizi committed
943
944
945
946
			assert_eq!(
				Auctions::auction_status(System::block_number()),
				AuctionStatus::<u32>::StartingPeriod
			);
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
		});
	}

	#[test]
	fn bidding_works() {
		new_test_ext().execute_with(|| {
			run_to_block(1);
			assert_ok!(Auctions::new_auction(Origin::signed(6), 5, 1));
			assert_ok!(Auctions::bid(Origin::signed(1), 0.into(), 1, 1, 4, 5));

			assert_eq!(Balances::reserved_balance(1), 5);
			assert_eq!(Balances::free_balance(1), 5);
			assert_eq!(
				Auctions::winning(0).unwrap()[SlotRange::ZeroThree as u8 as usize],
				Some((1, 0.into(), 5))
			);
		});
	}

	#[test]
	fn under_bidding_works() {
		new_test_ext().execute_with(|| {
			run_to_block(1);
			assert_ok!(Auctions::new_auction(Origin::signed(6), 5, 1));

			assert_ok!(Auctions::bid(Origin::signed(1), 0.into(), 1, 1, 4, 5));

Shawn Tabrizi's avatar
Shawn Tabrizi committed
974
975
976
			assert_storage_noop!({
				assert_ok!(Auctions::bid(Origin::signed(2), 0.into(), 1, 1, 4, 1));
			});
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
		});
	}

	#[test]
	fn over_bidding_works() {
		new_test_ext().execute_with(|| {
			run_to_block(1);
			assert_ok!(Auctions::new_auction(Origin::signed(6), 5, 1));
			assert_ok!(Auctions::bid(Origin::signed(1), 0.into(), 1, 1, 4, 5));
			assert_ok!(Auctions::bid(Origin::signed(2), 0.into(), 1, 1, 4, 6));

			assert_eq!(Balances::reserved_balance(1), 0);
			assert_eq!(Balances::free_balance(1), 10);
			assert_eq!(Balances::reserved_balance(2), 6);
			assert_eq!(Balances::free_balance(2), 14);
			assert_eq!(
				Auctions::winning(0).unwrap()[SlotRange::ZeroThree as u8 as usize],
				Some((2, 0.into(), 6))
			);
		});
	}

	#[test]
	fn auction_proceeds_correctly() {
For faster browsing, not all history is shown. View entire blame