From ec9dc2a110292c0787e69c6f8e93f528f6b69d9c Mon Sep 17 00:00:00 2001 From: soaresa <10797037+soaresa@users.noreply.github.com> Date: Mon, 5 Aug 2024 15:26:17 -0300 Subject: [PATCH] [move] PoF tiebreaker randomize tied bids, maintain fewer seats than bidders (replay PR) (#299) --- .gitignore | 8 +- .../src/libra_framework_sdk_builder.rs | 30 +- framework/drop-user-tools/jail_with_gc.move | 32 +- .../libra-framework/sources/genesis.move | 2 + .../sources/ol_sources/address_utils.move | 229 ++++++ .../sources/ol_sources/ancestry.move | 2 +- .../sources/ol_sources/epoch_boundary.move | 766 +++++++++--------- .../sources/ol_sources/infra_escrow.move | 11 +- .../sources/ol_sources/jail.move | 33 +- .../sources/ol_sources/mock.move | 546 +++++++------ .../sources/ol_sources/musical_chairs.move | 495 +++++------ .../sources/ol_sources/pledge_accounts.move | 88 +- .../sources/ol_sources/proof_of_fee.move | 720 +++++++++++----- .../sources/ol_sources/rank.todo | 2 +- .../ol_sources/tests/boundary.test.move | 201 ++++- .../sources/ol_sources/tests/burn.test.move | 553 +++++++------ .../ol_sources/tests/donor_voice.test.move | 13 +- .../sources/ol_sources/tests/jail.test.move | 3 +- .../ol_sources/tests/musical_chairs.test.move | 86 +- .../ol_sources/tests/proof_of_fee.test.move | 73 +- .../ol_sources/tests/slow_wallet.test.move | 9 +- .../sources/ol_sources/tests/stake.test.move | 24 +- .../tests/validator_reward.test.move | 131 +-- .../libra-framework/sources/randomness.move | 2 +- framework/libra-framework/sources/stake.move | 29 +- framework/releases/head.mrb | Bin 863549 -> 878837 bytes tools/config/src/make_yaml_public_fullnode.rs | 38 +- 27 files changed, 2452 insertions(+), 1674 deletions(-) create mode 100644 framework/libra-framework/sources/ol_sources/address_utils.move diff --git a/.gitignore b/.gitignore index 21446d574..1a8778d90 100644 --- a/.gitignore +++ b/.gitignore @@ -1,6 +1,6 @@ # Generated by Cargo # will have compiled files and executables -/target/ +**/target/ # Remove Cargo.lock from gitignore if creating an executable, leave it for libraries # More information here https://doc.rust-lang.org/cargo/guide/cargo-toml-vs-cargo-lock.html @@ -24,4 +24,8 @@ sccache.log *.bpl # exclude diem dependency -diem/ \ No newline at end of file +diem/ + +# exclude move coverage +.coverage_map.mvcov +.trace \ No newline at end of file diff --git a/framework/cached-packages/src/libra_framework_sdk_builder.rs b/framework/cached-packages/src/libra_framework_sdk_builder.rs index 4458a648e..83266963c 100644 --- a/framework/cached-packages/src/libra_framework_sdk_builder.rs +++ b/framework/cached-packages/src/libra_framework_sdk_builder.rs @@ -32,6 +32,8 @@ type Bytes = Vec; /// } /// ``` #[derive(Clone, Debug, PartialEq, Eq)] +#[cfg_attr(feature = "fuzzing", derive(proptest_derive::Arbitrary))] +#[cfg_attr(feature = "fuzzing", proptest(no_params))] pub enum EntryFunctionCall { /// Offers rotation capability on behalf of `account` to the account at address `recipient_address`. /// An account can delegate its rotation capability to only one other address at one time. If the account @@ -93,14 +95,14 @@ pub enum EntryFunctionCall { /// Generic authentication key rotation function that allows the user to rotate their authentication key from any scheme to any scheme. /// To authorize the rotation, we need two signatures: /// - the first signature `cap_rotate_key` refers to the signature by the account owner's current key on a valid `RotationProofChallenge`, - /// demonstrating that the user intends to and has the capability to rotate the authentication key of this account; + /// demonstrating that the user intends to and has the capability to rotate the authentication key of this account; /// - the second signature `cap_update_table` refers to the signature by the new key (that the account owner wants to rotate to) on a - /// valid `RotationProofChallenge`, demonstrating that the user owns the new private key, and has the authority to update the - /// `OriginatingAddress` map with the new address mapping ``. - /// To verify these two signatures, we need their corresponding public key and public key scheme: we use `from_scheme` and `from_public_key_bytes` - /// to verify `cap_rotate_key`, and `to_scheme` and `to_public_key_bytes` to verify `cap_update_table`. - /// A scheme of 0 refers to an Ed25519 key and a scheme of 1 refers to Multi-Ed25519 keys. - /// `originating address` refers to an account's original/first address. + /// valid `RotationProofChallenge`, demonstrating that the user owns the new private key, and has the authority to update the + /// `OriginatingAddress` map with the new address mapping ``. + /// To verify these two signatures, we need their corresponding public key and public key scheme: we use `from_scheme` and `from_public_key_bytes` + /// to verify `cap_rotate_key`, and `to_scheme` and `to_public_key_bytes` to verify `cap_update_table`. + /// A scheme of 0 refers to an Ed25519 key and a scheme of 1 refers to Multi-Ed25519 keys. + /// `originating address` refers to an account's original/first address. /// /// Here is an example attack if we don't ask for the second signature `cap_update_table`: /// Alice has rotated her account `addr_a` to `new_addr_a`. As a result, the following entry is created, to help Alice when recovering her wallet: @@ -1027,14 +1029,14 @@ pub fn account_revoke_signer_capability( /// Generic authentication key rotation function that allows the user to rotate their authentication key from any scheme to any scheme. /// To authorize the rotation, we need two signatures: /// - the first signature `cap_rotate_key` refers to the signature by the account owner's current key on a valid `RotationProofChallenge`, -/// demonstrating that the user intends to and has the capability to rotate the authentication key of this account; +/// demonstrating that the user intends to and has the capability to rotate the authentication key of this account; /// - the second signature `cap_update_table` refers to the signature by the new key (that the account owner wants to rotate to) on a -/// valid `RotationProofChallenge`, demonstrating that the user owns the new private key, and has the authority to update the -/// `OriginatingAddress` map with the new address mapping ``. -/// To verify these two signatures, we need their corresponding public key and public key scheme: we use `from_scheme` and `from_public_key_bytes` -/// to verify `cap_rotate_key`, and `to_scheme` and `to_public_key_bytes` to verify `cap_update_table`. -/// A scheme of 0 refers to an Ed25519 key and a scheme of 1 refers to Multi-Ed25519 keys. -/// `originating address` refers to an account's original/first address. +/// valid `RotationProofChallenge`, demonstrating that the user owns the new private key, and has the authority to update the +/// `OriginatingAddress` map with the new address mapping ``. +/// To verify these two signatures, we need their corresponding public key and public key scheme: we use `from_scheme` and `from_public_key_bytes` +/// to verify `cap_rotate_key`, and `to_scheme` and `to_public_key_bytes` to verify `cap_update_table`. +/// A scheme of 0 refers to an Ed25519 key and a scheme of 1 refers to Multi-Ed25519 keys. +/// `originating address` refers to an account's original/first address. /// /// Here is an example attack if we don't ask for the second signature `cap_update_table`: /// Alice has rotated her account `addr_a` to `new_addr_a`. As a result, the following entry is created, to help Alice when recovering her wallet: diff --git a/framework/drop-user-tools/jail_with_gc.move b/framework/drop-user-tools/jail_with_gc.move index cab301d38..1e355fcd0 100644 --- a/framework/drop-user-tools/jail_with_gc.move +++ b/framework/drop-user-tools/jail_with_gc.move @@ -160,31 +160,21 @@ module ol_framework::jail { }; } - /// gets a list of validators based on their jail reputation + /// Sort validators based on their jail reputation /// this is used in the bidding process for Proof-of-Fee where /// we seat the validators with the least amount of consecutive failures /// to rejoin. public(friend) fun sort_by_jail(vec_address: vector
): vector
acquires Jail { - - // Sorting the accounts vector based on value (weights). - // Bubble sort algorithm - let length = vector::length(&vec_address); - - let i = 0; - while (i < length){ - let j = 0; - while(j < length-i-1){ - - let (_, value_j) = get_jail_reputation(*vector::borrow(&vec_address, j)); - let (_, value_jp1) = get_jail_reputation(*vector::borrow(&vec_address, j + 1)); - - if(value_j > value_jp1){ - vector::swap
(&mut vec_address, j, j+1); - }; - j = j + 1; - }; - i = i + 1; - }; + // get the reputation of the validators + let reputation = vector::map(vec_address, |addr| { + let (_, value) = get_jail_reputation(addr); + value + }); + + // sort the validators based on their reputation + address_utils::sort_by_values(&mut vec_address, &mut reputation); + // shuffle duplicates scores to ensure fairness + address_utils::shuffle_duplicates(&mut vec_address, &reputation); vec_address } diff --git a/framework/libra-framework/sources/genesis.move b/framework/libra-framework/sources/genesis.move index 161e92c1b..f61594a56 100644 --- a/framework/libra-framework/sources/genesis.move +++ b/framework/libra-framework/sources/genesis.move @@ -24,6 +24,7 @@ module diem_framework::genesis { use diem_framework::transaction_fee; use diem_framework::transaction_validation; use diem_framework::version; + use diem_framework::randomness; //////// 0L //////// use diem_framework::validator_universe; @@ -137,6 +138,7 @@ module diem_framework::genesis { reconfiguration::initialize(&diem_framework_account); block::initialize(&diem_framework_account, epoch_interval_microsecs); state_storage::initialize(&diem_framework_account); + randomness::initialize(&diem_framework_account); //////// 0L //////// diff --git a/framework/libra-framework/sources/ol_sources/address_utils.move b/framework/libra-framework/sources/ol_sources/address_utils.move new file mode 100644 index 000000000..dd696932a --- /dev/null +++ b/framework/libra-framework/sources/ol_sources/address_utils.move @@ -0,0 +1,229 @@ +/// This module provides utility functions for handling and manipulating vectors of +/// addresses and their corresponding values. + +module ol_framework::address_utils { + use std::error; + use std::vector; + use diem_framework::randomness; + + // Error code for different length of addresses and values + const EDIFFERENT_LENGTH: u64 = 1; + + // A, B, C, easy as 1, 2, 3 + + // Bubble sort addresses and corresponding values + public fun sort_by_values(addresses: &mut vector
, values: &mut vector) { + assert!(vector::length(addresses) == vector::length(values), error::invalid_argument(EDIFFERENT_LENGTH)); + let len = vector::length(values); + let i = 0; + while (i < len) { + let j = 0; + while (j < len - i - 1) { + let value_j = *vector::borrow(values, j); + let value_jp1 = *vector::borrow(values, j + 1); + if (value_j > value_jp1) { + vector::swap(values, j, j + 1); + vector::swap
(addresses, j, j + 1); + }; + j = j + 1; + }; + i = i + 1; + }; + } + + // Shuffle addresses with the same values to ensure randomness position + public fun shuffle_duplicates(addresses: &mut vector
, values: &vector) { + assert!(vector::length(addresses) == vector::length(values), error::invalid_argument(EDIFFERENT_LENGTH)); + let len = vector::length(values); + let i = 0; + while (i < len) { + let j = i + 1; + while (j < len && *vector::borrow(values, i) == *vector::borrow(values, j)) { + j = j + 1; + }; + if (j > i + 1) { + let slice_len = j - i; + let perm = randomness::permutation(slice_len); + let temp_addresses = vector::empty
(); + let k = 0; + while (k < slice_len) { + let pos = i + k; + vector::push_back(&mut temp_addresses, *vector::borrow(addresses, pos)); + k = k + 1; + }; + k = 0; + while (k < slice_len) { + let perm_pos = *vector::borrow(&perm, k); + *vector::borrow_mut(addresses, i + k) = *vector::borrow(&temp_addresses, perm_pos); + k = k + 1; + }; + }; + i = j; + }; + } + + // Bubble Sort tests + + #[test] + #[expected_failure(abort_code = 0x10001, location = ol_framework::address_utils)] + fun test_sort_by_values_different_lengths() { + let values = vector[1, 2, 3]; + let addresses = vector[@0x1, @0x2]; + // This should trigger an assertion error due to different lengths + sort_by_values(&mut addresses, &mut values); + } + + #[test] + fun test_sort_empty_vectors() { + let values: vector = vector::empty(); + let addresses: vector
= vector::empty(); + sort_by_values(&mut addresses, &mut values); + assert!(values == vector[], 10002); + assert!(addresses == vector[], 10003); + } + + #[test] + fun test_sort_single_element() { + let values = vector[10]; + let addresses = vector[@0x1]; + sort_by_values(&mut addresses, &mut values); + assert!(values == vector[10], 10004); + assert!(addresses == vector[@0x1], 10005); + } + + #[test] + fun test_sort_already_sorted() { + let values = vector[1, 2, 3, 4, 5]; + let addresses = vector[@0x1, @0x2, @0x3, @0x4, @0x5]; + sort_by_values(&mut addresses, &mut values); + assert!(values == vector[1, 2, 3, 4, 5], 10006); + assert!(addresses == vector[@0x1, @0x2, @0x3, @0x4, @0x5], 10007); + } + + #[test] + fun test_sort_reverse_order() { + let values = vector[5, 4, 3, 2, 1]; + let addresses = vector[@0x5, @0x4, @0x3, @0x2, @0x1]; + sort_by_values(&mut addresses, &mut values); + assert!(values == vector[1, 2, 3, 4, 5], 10008); + assert!(addresses == vector[@0x1, @0x2, @0x3, @0x4, @0x5], 10009); + } + + #[test] + fun test_sort_with_duplicates() { + let values = vector[4, 2, 2, 3, 1]; + let addresses = vector[@0x1, @0x2, @0x3, @0x4, @0x5]; + sort_by_values(&mut addresses, &mut values); + assert!(values == vector[1, 2, 2, 3, 4], 10010); + assert!(addresses == vector[@0x5, @0x2, @0x3, @0x4, @0x1], 10011); + } + + #[test] + fun test_sort_random_order() { + let values = vector[3, 1, 4, 5, 2]; + let addresses = vector[@0x1, @0x2, @0x3, @0x4, @0x5]; + sort_by_values(&mut addresses, &mut values); + assert!(values == vector[1, 2, 3, 4, 5], 10012); + assert!(addresses == vector[@0x2, @0x5, @0x1, @0x3, @0x4], 10013); + } + + #[test] + fun test_sort_all_elements_equal() { + let values = vector[3, 3, 3, 3, 3]; + let addresses = vector[@0x1, @0x2, @0x3, @0x4, @0x5]; + sort_by_values(&mut addresses, &mut values); + assert!(values == vector[3, 3, 3, 3, 3], 10014); + assert!(addresses == vector[@0x1, @0x2, @0x3, @0x4, @0x5], 10015); + } + + + // Shuffle Tests + + #[test] + #[expected_failure(abort_code = 0x10001, location = ol_framework::address_utils)] + fun test_shuffle_duplicates_different_lengths() { + let values = vector[1, 2, 3]; + let addresses = vector[@0x1, @0x2]; + // This should trigger an assertion error due to different lengths + shuffle_duplicates(&mut addresses, &mut values); + } + + #[test] + fun test_shuffle_no_duplicates() { + // No duplicates in the values vector + let values = vector[1, 2, 3, 4, 5]; + let addresses = vector[@0x1, @0x2, @0x3, @0x4, @0x5]; + shuffle_duplicates(&mut addresses, &mut values); + assert!(values == vector[1, 2, 3, 4, 5], 10017); + assert!(addresses == vector[@0x1, @0x2, @0x3, @0x4, @0x5], 10018); + } + + #[test(root = @ol_framework)] + fun test_shuffle_with_duplicates(root: &signer) { + // One group of duplicates in the values vector + randomness::initialize_for_testing(root); + let values = vector[1, 2, 2, 3, 4]; + let addresses = vector[@0x1, @0x2, @0x3, @0x4, @0x5]; + let original_addresses = vector[@0x1, @0x2, @0x3, @0x4, @0x5]; + let shuffled = false; + let i = 0; + + while (i < 10) { + shuffle_duplicates(&mut addresses, &mut values); + if (addresses != original_addresses) { + shuffled = true; + break + }; + i = i + 1; + }; + + assert!(values == vector[1, 2, 2, 3, 4], 10019); + assert!(shuffled, 10020); + } + + #[test(root = @ol_framework)] + fun test_shuffle_multiple_duplicate_groups(root: &signer) { + // Multiple groups of duplicates in the values vector + randomness::initialize_for_testing(root); + let values = vector[1, 2, 2, 3, 3, 4]; + let addresses = vector[@0x1, @0x2, @0x3, @0x4, @0x5, @0x6]; + let original_addresses = vector[@0x1, @0x2, @0x3, @0x4, @0x5, @0x6]; + let shuffled = false; + let i = 0; + + while (i < 10) { + shuffle_duplicates(&mut addresses, &mut values); + if (addresses != original_addresses) { + shuffled = true; + break + }; + i = i + 1; + }; + + assert!(values == vector[1, 2, 2, 3, 3, 4], 10021); + assert!(shuffled, 10022); + } + + #[test(root = @ol_framework)] + fun test_shuffle_all_elements_equal(root: &signer) { + // All elements in the values vector are the same + randomness::initialize_for_testing(root); + let values = vector[2, 2, 2, 2]; + let addresses = vector[@0x1, @0x2, @0x3, @0x4]; + let original_addresses = vector[@0x1, @0x2, @0x3, @0x4]; + let shuffled = false; + let i = 0; + + while (i < 10) { + shuffle_duplicates(&mut addresses, &mut values); + if (addresses != original_addresses) { + shuffled = true; + break + }; + i = i + 1; + }; + + assert!(values == vector[2, 2, 2, 2], 10023); + assert!(shuffled, 10024); + } +} diff --git a/framework/libra-framework/sources/ol_sources/ancestry.move b/framework/libra-framework/sources/ol_sources/ancestry.move index 3bed49f99..0bb89368c 100644 --- a/framework/libra-framework/sources/ol_sources/ancestry.move +++ b/framework/libra-framework/sources/ol_sources/ancestry.move @@ -174,7 +174,7 @@ module ol_framework::ancestry { // now loop through all the accounts again, and check if this target // account is related to anyone. - let k = 0; + let k = 0; while (k < vector::length
(&list)) { let comparison_acc = vector::borrow(&list, k); // skip if you're the same person diff --git a/framework/libra-framework/sources/ol_sources/epoch_boundary.move b/framework/libra-framework/sources/ol_sources/epoch_boundary.move index b8df79f4f..ba93fa84c 100644 --- a/framework/libra-framework/sources/ol_sources/epoch_boundary.move +++ b/framework/libra-framework/sources/ol_sources/epoch_boundary.move @@ -1,382 +1,386 @@ module diem_framework::epoch_boundary { - use ol_framework::slow_wallet; - use ol_framework::musical_chairs; - use ol_framework::proof_of_fee; - use ol_framework::stake; - use ol_framework::libra_coin::LibraCoin; - use ol_framework::rewards; - use ol_framework::jail; - use ol_framework::safe; - use ol_framework::burn; - use ol_framework::donor_voice_txs; - use ol_framework::fee_maker; - use ol_framework::infra_escrow; - use ol_framework::libra_coin; - use ol_framework::match_index; - use ol_framework::community_wallet_init; - use ol_framework::testnet; - - use diem_framework::account; - use diem_framework::reconfiguration; - use diem_framework::transaction_fee; - use diem_framework::system_addresses; - use diem_framework::coin::{Coin}; - use std::vector; - use std::error; - use std::string; - use std::signer; - use diem_framework::create_signer; - - use diem_std::debug::print; - - friend diem_framework::genesis; - friend diem_framework::diem_governance; - friend diem_framework::block; // for testnet only - - //////// ERRORS //////// - /// The transaction fee coin has not been initialized - const ETX_FEES_NOT_INITIALIZED: u64 = 0; - - /// Epoch trigger can only be called on mainnet or in smoketests - const ETRIGGER_EPOCH_UNAUTHORIZED: u64 = 1; - - /// Epoch is not ready for reconfiguration - const ETRIGGER_NOT_READY: u64 = 2; + use std::error; + use std::vector; + use std::string; + use std::signer; + use diem_framework::account; + use diem_framework::randomness; + use diem_framework::coin::{Coin}; + use diem_framework::create_signer; + use diem_framework::reconfiguration; + use diem_framework::transaction_fee; + use diem_framework::system_addresses; + use ol_framework::stake; + use ol_framework::jail; + use ol_framework::safe; + use ol_framework::burn; + use ol_framework::rewards; + use ol_framework::testnet; + use ol_framework::fee_maker; + use ol_framework::libra_coin; + use ol_framework::slow_wallet; + use ol_framework::match_index; + use ol_framework::proof_of_fee; + use ol_framework::infra_escrow; + use ol_framework::musical_chairs; + use ol_framework::donor_voice_txs; + use ol_framework::libra_coin::LibraCoin; + use ol_framework::community_wallet_init; + + use diem_std::debug::print; + + friend diem_framework::genesis; + friend diem_framework::diem_governance; + friend diem_framework::block; // for testnet only + + //////// ERRORS //////// + /// The transaction fee coin has not been initialized + const ETX_FEES_NOT_INITIALIZED: u64 = 0; + /// Epoch trigger can only be called on mainnet or in smoketests + const ETRIGGER_EPOCH_UNAUTHORIZED: u64 = 1; + /// Epoch is not ready for reconfiguration + const ETRIGGER_NOT_READY: u64 = 2; + /// Epoch number mismat + const ENOT_SAME_EPOCH: u64 = 3; + + /////// Constants //////// + /// How many PoF baseline rewards to we set aside for the miners. + /// equivalent reward of one seats of the validator set + const ORACLE_PROVIDERS_SEATS: u64 = 1; + + /// The VM can set the boundary bit to allow reconfiguration + struct BoundaryBit has key { + ready: bool, + closing_epoch: u64, + } - /// Epoch number mismat - const ENOT_SAME_EPOCH: u64 = 3; + /// I just checked in, to see what condition my condition was in. + struct BoundaryStatus has key, drop { + security_bill_count: u64, + security_bill_amount: u64, + security_bill_success: bool, + + dd_accounts_count: u64, + dd_accounts_amount: u64, + dd_accounts_success: bool, + + set_fee_makers_success: bool, + system_fees_collected: u64, + // Process Outgoing + outgoing_vals_paid: vector
, + outgoing_total_reward: u64, + outgoing_nominal_reward_to_vals: u64, + outgoing_entry_fee: u64, + outgoing_clearing_percent: u64, + outgoing_vals_success: bool, // TODO + + // Oracle / Tower + tower_state_success: bool, // TODO + oracle_budget: u64, + oracle_pay_count: u64, + oracle_pay_amount: u64, + oracle_pay_success: bool, + + epoch_burn_fees: u64, + epoch_burn_success: bool, + + slow_wallet_drip_amount: u64, + slow_wallet_drip_success: bool, + // Process Incoming + // musical chairs + incoming_compliant: vector
, + incoming_compliant_count: u64, + incoming_seats_offered: u64, + + // proof of fee + incoming_all_bidders: vector
, + incoming_only_qualified_bidders: vector
, + incoming_auction_winners: vector
, + incoming_filled_seats: u64, + incoming_fees: u64, + incoming_fees_success: bool, + + // reconfiguration + incoming_post_failover_check: vector
, + incoming_vals_missing_configs: vector
, + incoming_actual_vals: vector
, + incoming_final_set_size: u64, + incoming_reconfig_success: bool, + + infra_subsidize_amount: u64, // TODO + infra_subsidize_success: bool, // TODO + + pof_thermo_success: bool, + pof_thermo_increase: bool, + pof_thermo_amount: u64, + } - /////// Constants //////// - /// how many PoF baseline rewards to we set aside for the miners. - /// equivalent reward of one seats of the validator set - const ORACLE_PROVIDERS_SEATS: u64 = 1; + /// initialize structs, requires both signers since BoundaryBit can only be + // accessed by VM + public(friend) fun initialize(framework_signer: &signer) { + if (!exists(@ol_framework)){ + move_to(framework_signer, reset()); + }; - // the VM can set the boundary bit to allow reconfiguration - struct BoundaryBit has key { - ready: bool, - closing_epoch: u64, + // boundary bit can only be written by VM + if (!exists(@ol_framework)) { + move_to(framework_signer, BoundaryBit { + ready: false, + closing_epoch: 0, + }); } + } - // I just checked in, to see what condition my condition was in. - struct BoundaryStatus has key, drop { - security_bill_count: u64, - security_bill_amount: u64, - security_bill_success: bool, + fun reset(): BoundaryStatus { + BoundaryStatus { + security_bill_count: 0, + security_bill_amount: 0, + security_bill_success: false, + + dd_accounts_count: 0, + dd_accounts_amount: 0, + dd_accounts_success: false, - dd_accounts_count: u64, - dd_accounts_amount: u64, - dd_accounts_success: bool, + set_fee_makers_success: false, + system_fees_collected: 0, - set_fee_makers_success: bool, - system_fees_collected: u64, // Process Outgoing - outgoing_vals_paid: vector
, - outgoing_total_reward: u64, - outgoing_nominal_reward_to_vals: u64, - outgoing_entry_fee: u64, - outgoing_clearing_percent: u64, - outgoing_vals_success: bool, // TODO + outgoing_vals_paid: vector::empty(), + outgoing_total_reward: 0, + outgoing_vals_success: false, + outgoing_nominal_reward_to_vals: 0, + outgoing_entry_fee: 0, + outgoing_clearing_percent: 0, // Oracle / Tower - tower_state_success: bool, // TODO - oracle_budget: u64, - oracle_pay_count: u64, - oracle_pay_amount: u64, - oracle_pay_success: bool, - - epoch_burn_fees: u64, - epoch_burn_success: bool, - - slow_wallet_drip_amount: u64, - slow_wallet_drip_success: bool, + tower_state_success: false, + oracle_budget: 0, + oracle_pay_count: 0, + oracle_pay_amount: 0, + oracle_pay_success: false, + epoch_burn_fees: 0, + epoch_burn_success: false, + + slow_wallet_drip_amount: 0, + slow_wallet_drip_success: false, // Process Incoming - // musical chairs - incoming_compliant: vector
, - incoming_compliant_count: u64, - incoming_seats_offered: u64, - - // proof of fee - incoming_all_bidders: vector
, - incoming_only_qualified_bidders: vector
, - incoming_auction_winners: vector
, - incoming_filled_seats: u64, - incoming_fees: u64, - incoming_fees_success: bool, - - // reconfiguration - incoming_post_failover_check: vector
, - incoming_vals_missing_configs: vector
, - incoming_actual_vals: vector
, - incoming_final_set_size: u64, - incoming_reconfig_success: bool, - - infra_subsidize_amount: u64, // TODO - infra_subsidize_success: bool, // TODO - - pof_thermo_success: bool, - pof_thermo_increase: bool, - pof_thermo_amount: u64, - } - - /// initialize structs, requires both signers since BoundaryBit can only be - // accessed by VM - public(friend) fun initialize(framework_signer: &signer) { - if (!exists(@ol_framework)){ - move_to(framework_signer, reset()); - }; - - // boundary bit can only be written by VM - if (!exists(@ol_framework)) { - move_to(framework_signer, BoundaryBit { - ready: false, - closing_epoch: 0, - }); - } + incoming_compliant: vector::empty(), + incoming_compliant_count: 0, + incoming_seats_offered: 0, + incoming_filled_seats: 0, + incoming_auction_winners: vector::empty(), + incoming_all_bidders: vector::empty(), + incoming_only_qualified_bidders: vector::empty(), + incoming_final_set_size: 0, + incoming_fees: 0, + incoming_fees_success: false, + + incoming_post_failover_check: vector::empty(), + incoming_vals_missing_configs: vector::empty(), + incoming_actual_vals: vector::empty(), + incoming_reconfig_success: false, + + infra_subsidize_amount: 0, + infra_subsidize_success: false, + + pof_thermo_success: false, + pof_thermo_increase: false, + pof_thermo_amount: 0, } + } - fun reset(): BoundaryStatus { - BoundaryStatus { - security_bill_count: 0, - security_bill_amount: 0, - security_bill_success: false, - - dd_accounts_count: 0, - dd_accounts_amount: 0, - dd_accounts_success: false, - - set_fee_makers_success: false, - system_fees_collected: 0, - - // Process Outgoing - outgoing_vals_paid: vector::empty(), - outgoing_total_reward: 0, - outgoing_vals_success: false, - outgoing_nominal_reward_to_vals: 0, - outgoing_entry_fee: 0, - outgoing_clearing_percent: 0, - - // Oracle / Tower - tower_state_success: false, - oracle_budget: 0, - oracle_pay_count: 0, - oracle_pay_amount: 0, - oracle_pay_success: false, - epoch_burn_fees: 0, - epoch_burn_success: false, - - slow_wallet_drip_amount: 0, - slow_wallet_drip_success: false, - // Process Incoming - incoming_compliant: vector::empty(), - incoming_compliant_count: 0, - incoming_seats_offered: 0, - incoming_filled_seats: 0, - incoming_auction_winners: vector::empty(), - incoming_all_bidders: vector::empty(), - incoming_only_qualified_bidders: vector::empty(), - incoming_final_set_size: 0, - incoming_fees: 0, - incoming_fees_success: false, - - incoming_post_failover_check: vector::empty(), - incoming_vals_missing_configs: vector::empty(), - incoming_actual_vals: vector::empty(), - incoming_reconfig_success: false, - - infra_subsidize_amount: 0, - infra_subsidize_success: false, - - pof_thermo_success: false, - pof_thermo_increase: false, - pof_thermo_amount: 0, - } + ///TODO: epoch trigger is currently disabled and requires further testing. + /// refer to block.move and std::features + /// flip the bit to allow the epoch to be reconfigured on any transaction + public(friend) fun enable_epoch_trigger(vm_signer: &signer, closing_epoch: + u64) acquires BoundaryBit { + if (signer::address_of(vm_signer) != @vm_reserved) return; + + // though the VM is calling this, we need the struct to be on + // diem_framework. So we need to use create_signer + + let framework_signer = create_signer::create_signer(@ol_framework); + if (!exists(@ol_framework)) { + // Just like a prayer, your voice can take me there + // Just like a muse to me, you are a mystery + // Just like a dream, you are not what you seem + // Just like a prayer, no choice your voice can take me there... + move_to(&framework_signer, BoundaryBit { + closing_epoch: closing_epoch, + ready: true, + }) + } else { + let state = borrow_global_mut(@ol_framework); + state.closing_epoch = closing_epoch; + state.ready = true; } + } - ///TODO: epoch trigger is currently disabled and requires further testing. - /// refer to block.move and std::features - /// flip the bit to allow the epoch to be reconfigured on any transaction - public(friend) fun enable_epoch_trigger(vm_signer: &signer, closing_epoch: - u64) acquires BoundaryBit { - if (signer::address_of(vm_signer) != @vm_reserved) return; - - // though the VM is calling this, we need the struct to be on - // diem_framework. So we need to use create_signer - - let framework_signer = create_signer::create_signer(@ol_framework); - if (!exists(@ol_framework)) { - // Just like a prayer, your voice can take me there - // Just like a muse to me, you are a mystery - // Just like a dream, you are not what you seem - // Just like a prayer, no choice your voice can take me there... - move_to(&framework_signer, BoundaryBit { - closing_epoch: closing_epoch, - ready: true, - }) - } else { - let state = borrow_global_mut(@ol_framework); - state.closing_epoch = closing_epoch; - state.ready = true; - } - } - /// Once epoch boundary time has passed, and the BoundaryBit set to true - /// any user can trigger the epoch boundary. - /// Why do this? It's preferable that the VM never trigger any function. - /// An abort by the VM will cause a network halt. The same abort, if called - /// by a user, would not cause a halt. - public(friend) fun trigger_epoch(framework_signer: &signer) acquires BoundaryBit, - BoundaryStatus { - // COMMIT NOTE: there's no reason to gate this, if th trigger is not - // ready (which only happens on Main and Stage, then user will get an error) - // assert!(!testnet::is_testnet(), ETRIGGER_EPOCH_MAINNET); - // must get root permission from governance.move - system_addresses::assert_diem_framework(framework_signer); - let _ = can_trigger(); // will abort if false - - // update the state and flip the Bit - // note we are using the 0x0 address for BoundaryBit - let state = borrow_global_mut(@ol_framework); - state.ready = false; + /// Once epoch boundary time has passed, and the BoundaryBit set to true + /// any user can trigger the epoch boundary. + /// Why do this? It's preferable that the VM never trigger any function. + /// An abort by the VM will cause a network halt. The same abort, if called + /// by a user, would not cause a halt. + public(friend) fun trigger_epoch(framework_signer: &signer) + acquires BoundaryBit, BoundaryStatus { + // COMMIT NOTE: there's no reason to gate this, if th trigger is not + // ready (which only happens on Main and Stage, then user will get an error) + // assert!(!testnet::is_testnet(), ETRIGGER_EPOCH_MAINNET); + // must get root permission from governance.move + system_addresses::assert_diem_framework(framework_signer); + let _ = can_trigger(); // will abort if false + + // update the state and flip the Bit + // note we are using the 0x0 address for BoundaryBit + let state = borrow_global_mut(@ol_framework); + state.ready = false; + + epoch_boundary(framework_signer, state.closing_epoch, 0); + } - epoch_boundary(framework_signer, state.closing_epoch, 0); - } + // utility to use in smoke tests + public entry fun smoke_trigger_epoch(framework_signer: &signer) + acquires BoundaryBit, BoundaryStatus { + // cannot call this on mainnet + // only for smoke testing + assert!(testnet::is_not_mainnet(), ETRIGGER_EPOCH_UNAUTHORIZED); + // must get 0x1 sig from governance.move + system_addresses::assert_diem_framework(framework_signer); + let state = borrow_global_mut(@ol_framework); + epoch_boundary(framework_signer, state.closing_epoch, 0); + } - // utility to use in smoke tests - public entry fun smoke_trigger_epoch(framework_signer: &signer) acquires BoundaryBit, - BoundaryStatus { - // cannot call this on mainnet - // only for smoke testing - assert!(testnet::is_not_mainnet(), ETRIGGER_EPOCH_UNAUTHORIZED); - // must get 0x1 sig from governance.move - system_addresses::assert_diem_framework(framework_signer); - let state = borrow_global_mut(@ol_framework); - epoch_boundary(framework_signer, state.closing_epoch, 0); - } + #[view] + /// check to see if the epoch Boundary Bit is true + public fun can_trigger(): bool acquires BoundaryBit { + let state = borrow_global_mut(@ol_framework); + assert!(state.ready, ETRIGGER_NOT_READY); + assert!(state.closing_epoch == reconfiguration::get_current_epoch(), + ENOT_SAME_EPOCH); + true + } - #[view] - /// check to see if the epoch Boundary Bit is true - public fun can_trigger(): bool acquires BoundaryBit { - let state = borrow_global_mut(@ol_framework); - assert!(state.ready, ETRIGGER_NOT_READY); - assert!(state.closing_epoch == reconfiguration::get_current_epoch(), - ENOT_SAME_EPOCH); - true - } + // This function handles the necessary migrations that occur at the epoch boundary + // when new modules or structures are added by chain upgrades. + fun migrate_data(root: &signer) { + randomness::initialize(root); + } - // Contains all of 0L's business logic for end of epoch. - // This removed business logic from reconfiguration.move - // and prevents dependency cycling. - public(friend) fun epoch_boundary(root: &signer, closing_epoch: u64, - epoch_round: u64) acquires BoundaryStatus { - print(&string::utf8(b"EPOCH BOUNDARY BEGINS")); - // either 0x0 or 0x1 can call, but we will always use framework signer - system_addresses::assert_ol(root); - let root = &create_signer::create_signer(@ol_framework); - let status = borrow_global_mut(@ol_framework); - - print(&string::utf8(b"status reset")); - *status = reset(); - - // bill root service fees; - print(&string::utf8(b"root_service_billing")); - root_service_billing(root, status); - - print(&string::utf8(b"process_donor_voice_accounts")); - // run the transactions of donor directed accounts - let (count, amount, success) = donor_voice_txs::process_donor_voice_accounts(root, closing_epoch); - status.dd_accounts_count = count; - status.dd_accounts_amount = amount; - status.dd_accounts_success = success; - - print(&string::utf8(b"tower_state::reconfig")); - // reset fee makers tracking - status.set_fee_makers_success = fee_maker::epoch_reset_fee_maker(root); - - print(&string::utf8(b"musical_chairs::stop_the_music")); - let (compliant_vals, n_seats) = musical_chairs::stop_the_music(root, - closing_epoch, epoch_round); - status.incoming_compliant_count = vector::length(&compliant_vals); - status.incoming_compliant = compliant_vals; - status.incoming_seats_offered = n_seats; - - print(&string::utf8(b"settle_accounts")); - - settle_accounts(root, compliant_vals, status); - - print(&string::utf8(b"slow_wallet::on_new_epoch")); - // drip coins - let (s_success, s_amount) = slow_wallet::on_new_epoch(root); - - status.slow_wallet_drip_amount = s_amount; - status.slow_wallet_drip_success = s_success; - - // ======= THE BOUNDARY ======= - // And to know yourself - // is to be yourself - // keeps you walking through these tears. - - print(&string::utf8(b"process_incoming_validators")); - process_incoming_validators(root, status, compliant_vals, n_seats); - - print(&string::utf8(b"subsidize_from_infra_escrow")); - let (i_success, i_fee) = subsidize_from_infra_escrow(root); - status.infra_subsidize_amount = i_fee; - status.infra_subsidize_success = i_success; - - print(&string::utf8(b"reward_thermostat")); - let (t_success, t_increase, t_amount) = - proof_of_fee::reward_thermostat(root); - status.pof_thermo_success = t_success; - status.pof_thermo_increase = t_increase; - status.pof_thermo_amount = t_amount; - - print(&string::utf8(b"EPOCH BOUNDARY END")); - print(status); - reconfiguration::reconfigure(); + // Contains all of 0L's business logic for end of epoch. + // This removed business logic from reconfiguration.move + // and prevents dependency cycling. + public(friend) fun epoch_boundary(root: &signer, closing_epoch: u64, epoch_round: u64) + acquires BoundaryStatus { + print(&string::utf8(b"EPOCH BOUNDARY BEGINS")); + // either 0x0 or 0x1 can call, but we will always use framework signer + system_addresses::assert_ol(root); + let root = &create_signer::create_signer(@ol_framework); + let status = borrow_global_mut(@ol_framework); + + print(&string::utf8(b"migrate_data")); + migrate_data(root); + + print(&string::utf8(b"status reset")); + *status = reset(); + + // bill root service fees; + print(&string::utf8(b"root_service_billing")); + root_service_billing(root, status); + + print(&string::utf8(b"process_donor_voice_accounts")); + // run the transactions of donor directed accounts + let (count, amount, success) = donor_voice_txs::process_donor_voice_accounts(root, closing_epoch); + status.dd_accounts_count = count; + status.dd_accounts_amount = amount; + status.dd_accounts_success = success; + + print(&string::utf8(b"tower_state::reconfig")); + // reset fee makers tracking + status.set_fee_makers_success = fee_maker::epoch_reset_fee_maker(root); + + print(&string::utf8(b"musical_chairs::stop_the_music")); + let (compliant_vals, n_seats) = musical_chairs::stop_the_music(root, + closing_epoch, epoch_round); + status.incoming_compliant_count = vector::length(&compliant_vals); + status.incoming_compliant = compliant_vals; + status.incoming_seats_offered = n_seats; + + print(&string::utf8(b"settle_accounts")); + settle_accounts(root, compliant_vals, status); + + print(&string::utf8(b"slow_wallet::on_new_epoch")); + // drip coins + let (s_success, s_amount) = slow_wallet::on_new_epoch(root); + status.slow_wallet_drip_amount = s_amount; + status.slow_wallet_drip_success = s_success; + + // ======= THE BOUNDARY ======= + // And to know yourself + // is to be yourself + // keeps you walking through these tears. + + print(&string::utf8(b"process_incoming_validators")); + process_incoming_validators(root, status, compliant_vals, n_seats); + + print(&string::utf8(b"reward_thermostat")); + let (t_success, t_increase, t_amount) = + proof_of_fee::reward_thermostat(root); + status.pof_thermo_success = t_success; + status.pof_thermo_increase = t_increase; + status.pof_thermo_amount = t_amount; + + print(&string::utf8(b"subsidize_from_infra_escrow")); + let (i_success, i_fee) = subsidize_from_infra_escrow(root); + status.infra_subsidize_amount = i_fee; + status.infra_subsidize_success = i_success; + + print(&string::utf8(b"EPOCH BOUNDARY END")); + print(status); + reconfiguration::reconfigure(); } /// withdraw coins and settle accounts for validators and oracles /// returns the list of compliant_vals fun settle_accounts(root: &signer, compliant_vals: vector
, status: &mut BoundaryStatus): vector
{ - assert!(transaction_fee::is_fees_collection_enabled(), error::invalid_state(ETX_FEES_NOT_INITIALIZED)); - - if (transaction_fee::system_fees_collected() > 0) { - let all_fees = transaction_fee::root_withdraw_all(root); - status.system_fees_collected = libra_coin::value(&all_fees); - - // Nominal fee set by the PoF thermostat - let (nominal_reward_to_vals, entry_fee, clearing_percent, _ ) = proof_of_fee::get_consensus_reward(); - status.outgoing_nominal_reward_to_vals = nominal_reward_to_vals; - status.outgoing_entry_fee = entry_fee; - status.outgoing_clearing_percent = clearing_percent; - - // validators get the gross amount of the reward, since they already paid to enter. This results in a net payment equivalent to: - // nominal_reward_to_vals - entry_fee. - let (compliant_vals, total_reward) = process_outgoing_validators(root, &mut all_fees, nominal_reward_to_vals, compliant_vals); - - status.outgoing_vals_paid = compliant_vals; - status.outgoing_total_reward = total_reward; - - // check that the sustem fees collect were greater than reward - status.outgoing_vals_success = (status.system_fees_collected >= total_reward); - // check the the total actually deposited/paid is the expected amount - if (nominal_reward_to_vals > 0) { // check for zero - status.outgoing_vals_success = total_reward == (vector::length(&compliant_vals) * nominal_reward_to_vals) - }; - - // Commit note: deprecated with tower mining. - - // remainder gets burnt according to fee maker preferences - let (b_success, b_fees) = burn::epoch_burn_fees(root, &mut all_fees); - status.epoch_burn_success = b_success; - status.epoch_burn_fees = b_fees; - - // coin can finally be destroyed. Up to here we have been extracting from a mutable. - // It's possible there might be some dust, that should get burned - burn::burn_and_track(all_fees); - }; + assert!(transaction_fee::is_fees_collection_enabled(), error::invalid_state(ETX_FEES_NOT_INITIALIZED)); + + if (transaction_fee::system_fees_collected() > 0) { + let all_fees = transaction_fee::root_withdraw_all(root); + status.system_fees_collected = libra_coin::value(&all_fees); + + // Nominal fee set by the PoF thermostat + let (nominal_reward_to_vals, entry_fee, clearing_percent, _ ) = proof_of_fee::get_consensus_reward(); + status.outgoing_nominal_reward_to_vals = nominal_reward_to_vals; + status.outgoing_entry_fee = entry_fee; + status.outgoing_clearing_percent = clearing_percent; + + // validators get the gross amount of the reward, since they already paid to enter. This results in a net payment equivalent to: + // nominal_reward_to_vals - entry_fee. + let (compliant_vals, total_reward) = process_outgoing_validators(root, &mut all_fees, nominal_reward_to_vals, compliant_vals); + status.outgoing_vals_paid = compliant_vals; + status.outgoing_total_reward = total_reward; + + // check that the system fees collect were greater than reward + status.outgoing_vals_success = (status.system_fees_collected >= total_reward); + // check the the total actually deposited/paid is the expected amount + if (nominal_reward_to_vals > 0) { // check for zero + status.outgoing_vals_success = total_reward == (vector::length(&compliant_vals) * nominal_reward_to_vals) + }; - compliant_vals + // Commit note: deprecated with tower mining. + + // remainder gets burnt according to fee maker preferences + let (b_success, b_fees) = burn::epoch_burn_fees(root, &mut all_fees); + status.epoch_burn_success = b_success; + status.epoch_burn_fees = b_fees; + + // coin can finally be destroyed. Up to here we have been extracting from a mutable. + // It's possible there might be some dust, that should get burned + burn::burn_and_track(all_fees); + }; + + compliant_vals } @@ -403,11 +407,11 @@ module diem_framework::epoch_boundary { jail::jail(root, *addr); } else { // vector::push_back(&mut compliant_vals, *addr); - if (libra_coin::value(reward_budget) > reward_per) { + if (libra_coin::value(reward_budget) >= reward_per) { let user_coin = libra_coin::extract(reward_budget, reward_per); reward_deposited = reward_deposited + libra_coin::value(&user_coin); rewards::process_single(root, *addr, user_coin, 1); - } + }; }; i = i + 1; @@ -427,7 +431,6 @@ module diem_framework::epoch_boundary { status.incoming_only_qualified_bidders = only_qualified_bidders; status.incoming_auction_winners = auction_winners; - let post_failover_check = stake::check_failover_rules(auction_winners, n_seats); status.incoming_post_failover_check = post_failover_check; @@ -449,25 +452,22 @@ module diem_framework::epoch_boundary { // set up rewards subsidy for coming epoch fun subsidize_from_infra_escrow(root: &signer): (bool, u64) { - system_addresses::assert_ol(root); - let (reward_per, _, _, _ ) = proof_of_fee::get_consensus_reward(); - let vals = stake::get_current_validators(); - let count_vals = vector::length(&vals); - count_vals = count_vals + ORACLE_PROVIDERS_SEATS; - let total_epoch_budget = count_vals * reward_per; - infra_escrow::epoch_boundary_collection(root, - total_epoch_budget) + system_addresses::assert_ol(root); + let (reward_per, _, _, _ ) = proof_of_fee::get_consensus_reward(); + let vals = stake::get_current_validators(); + let count_vals = vector::length(&vals); + let total_epoch_budget = (count_vals * reward_per) + 1; // +1 for rounding + infra_escrow::epoch_boundary_collection(root, total_epoch_budget) } - /// check qualifications of community wallets - /// need to check every epoch so that wallets who no longer qualify are not biasing the Match algorithm. - fun reset_match_index_ratios(root: &signer) { - system_addresses::assert_ol(root); - let list = match_index::get_address_list(); - let good = community_wallet_init::get_qualifying(list); - match_index::calc_ratios(root, good); - } - + /// check qualifications of community wallets + /// need to check every epoch so that wallets who no longer qualify are not biasing the Match algorithm. + fun reset_match_index_ratios(root: &signer) { + system_addresses::assert_ol(root); + let list = match_index::get_address_list(); + let good = community_wallet_init::get_qualifying(list); + match_index::calc_ratios(root, good); + } // all services the root collective security is billing for fun root_service_billing(vm: &signer, status: &mut BoundaryStatus) { @@ -505,25 +505,27 @@ module diem_framework::epoch_boundary { #[test_only] public fun ol_reconfigure_for_test(vm: &signer, closing_epoch: u64, epoch_round: u64) acquires BoundaryStatus { - use diem_framework::system_addresses; + use diem_framework::system_addresses; + use diem_framework::randomness; - system_addresses::assert_ol(vm); - epoch_boundary(vm, closing_epoch, epoch_round); + system_addresses::assert_ol(vm); + randomness::initialize_for_testing(vm); + epoch_boundary(vm, closing_epoch, epoch_round); } #[test_only] public fun test_set_boundary_ready(framework: &signer, closing_epoch: u64) acquires BoundaryBit { - system_addresses::assert_ol(framework); - // don't check for "testnet" here, otherwise we can't test the production - // settings - let vm_signer = create_signer::create_signer(@vm_reserved); - enable_epoch_trigger(&vm_signer, closing_epoch); + system_addresses::assert_ol(framework); + // don't check for "testnet" here, otherwise we can't test the production + // settings + let vm_signer = create_signer::create_signer(@vm_reserved); + enable_epoch_trigger(&vm_signer, closing_epoch); } #[test_only] public fun test_trigger(vm: &signer) acquires BoundaryStatus, BoundaryBit { - // don't check for "testnet" here, otherwise we can't test the production settings - trigger_epoch(vm); + // don't check for "testnet" here, otherwise we can't test the production settings + trigger_epoch(vm); } } diff --git a/framework/libra-framework/sources/ol_sources/infra_escrow.move b/framework/libra-framework/sources/ol_sources/infra_escrow.move index c41061c5a..b91fbd34d 100644 --- a/framework/libra-framework/sources/ol_sources/infra_escrow.move +++ b/framework/libra-framework/sources/ol_sources/infra_escrow.move @@ -10,21 +10,21 @@ /////////////////////////////////////////////////////////////////////////// module ol_framework::infra_escrow{ + use std::error; use std::option::{Self, Option}; + use diem_framework::coin; + use diem_framework::transaction_fee; use diem_framework::system_addresses; + use ol_framework::ol_account; use ol_framework::libra_coin::LibraCoin; use ol_framework::pledge_accounts; // use ol_framework::slow_wallet; - use ol_framework::ol_account; - use diem_framework::coin; - use diem_framework::transaction_fee; // use std::fixed_point32; // use std::signer; - use std::error; // use diem_std::debug::print; - friend ol_framework::epoch_boundary; friend diem_framework::genesis; + friend ol_framework::epoch_boundary; const EGENESIS_REWARD: u64 = 0; /// for use on genesis, creates the infra escrow pledge policy struct @@ -71,7 +71,6 @@ module ol_framework::infra_escrow{ } - // Transaction script for user to pledge to infra escrow. fun user_pledge_infra(user_sig: &signer, amount: u64){ pledge_accounts::user_pledge(user_sig, @ol_framework, amount); diff --git a/framework/libra-framework/sources/ol_sources/jail.move b/framework/libra-framework/sources/ol_sources/jail.move index 8c32fd4a4..cde4d4e77 100644 --- a/framework/libra-framework/sources/ol_sources/jail.move +++ b/framework/libra-framework/sources/ol_sources/jail.move @@ -46,6 +46,7 @@ module ol_framework::jail { use std::error; use ol_framework::vouch; use ol_framework::stake; + use ol_framework::address_utils; friend ol_framework::validator_universe; friend ol_framework::epoch_boundary; @@ -157,31 +158,21 @@ module ol_framework::jail { }; } - /// gets a list of validators based on their jail reputation + /// Sort validators based on their jail reputation /// this is used in the bidding process for Proof-of-Fee where /// we seat the validators with the least amount of consecutive failures /// to rejoin. public(friend) fun sort_by_jail(vec_address: vector
): vector
acquires Jail { - - // Sorting the accounts vector based on value (weights). - // Bubble sort algorithm - let length = vector::length(&vec_address); - - let i = 0; - while (i < length){ - let j = 0; - while(j < length-i-1){ - - let (_, value_j) = get_jail_reputation(*vector::borrow(&vec_address, j)); - let (_, value_jp1) = get_jail_reputation(*vector::borrow(&vec_address, j + 1)); - - if(value_j > value_jp1){ - vector::swap
(&mut vec_address, j, j+1); - }; - j = j + 1; - }; - i = i + 1; - }; + // get the reputation of the validators + let reputation = vector::map(vec_address, |addr| { + let (_, value) = get_jail_reputation(addr); + value + }); + + // sort the validators based on their reputation + address_utils::sort_by_values(&mut vec_address, &mut reputation); + // shuffle duplicates scores to ensure fairness + address_utils::shuffle_duplicates(&mut vec_address, &reputation); vec_address } diff --git a/framework/libra-framework/sources/ol_sources/mock.move b/framework/libra-framework/sources/ol_sources/mock.move index b99e36079..40acbcecd 100644 --- a/framework/libra-framework/sources/ol_sources/mock.move +++ b/framework/libra-framework/sources/ol_sources/mock.move @@ -1,26 +1,28 @@ // Some fixtures are complex and are repeatedly needed #[test_only] module ol_framework::mock { + use std::vector; + use std::signer; + use diem_framework::coin; + use diem_framework::block; use diem_framework::stake; + use diem_framework::account; + use diem_framework::genesis; + use diem_framework::timestamp; use diem_framework::reconfiguration; + use diem_framework::system_addresses; + use diem_framework::transaction_fee; use ol_framework::grade; use ol_framework::vouch; - use std::vector; - use diem_framework::genesis; - use diem_framework::account; use ol_framework::slow_wallet; use ol_framework::proof_of_fee; use ol_framework::validator_universe; - use diem_framework::timestamp; - use diem_framework::system_addresses; use ol_framework::epoch_boundary; - use diem_framework::coin; use ol_framework::libra_coin::{Self, LibraCoin}; - use diem_framework::transaction_fee; use ol_framework::ol_account; use ol_framework::epoch_helper; use ol_framework::musical_chairs; - use diem_framework::block; + use ol_framework::pledge_accounts; // use diem_std::debug::print; @@ -36,265 +38,280 @@ module ol_framework::mock { #[test_only] public fun reset_val_perf_all(vm: &signer) { - let vals = stake::get_current_validators(); - let i = 0; - while (i < vector::length(&vals)) { - let a = vector::borrow(&vals, i); - stake::mock_performance(vm, *a, 0, 0); - i = i + 1; - }; + let vals = stake::get_current_validators(); + let i = 0; + while (i < vector::length(&vals)) { + let a = vector::borrow(&vals, i); + stake::mock_performance(vm, *a, 0, 0); + i = i + 1; + }; } #[test_only] - public fun mock_case_1(vm: &signer, addr: address){ - assert!(stake::is_valid(addr), 01); - stake::mock_performance(vm, addr, 100, 10); - let (compliant, _, _) = grade::get_validator_grade(addr); - assert!(compliant, 01); - } - - - #[test_only] - // did not do enough mining, but did validate. - public fun mock_case_4(vm: &signer, addr: address){ - assert!(stake::is_valid(addr), 01); - stake::mock_performance(vm, addr, 0, 100); // 100 failing proposals - // let (compliant, _, _, _) = grade::get_validator_grade(addr, 0); - let (compliant, _, _) = grade::get_validator_grade(addr); - assert!(!compliant, 02); - } - - // Mock all nodes being compliant case 1 - #[test_only] - public fun mock_all_vals_good_performance(vm: &signer) { - - let vals = stake::get_current_validators(); - - let i = 0; - while (i < vector::length(&vals)) { - - let a = vector::borrow(&vals, i); - mock_case_1(vm, *a); - i = i + 1; - }; - - } - - //////// PROOF OF FEE //////// - #[test_only] - public fun pof_default(): (vector
, vector, vector){ - - let vals = stake::get_current_validators(); - - let (bids, expiry) = mock_bids(&vals); - - // make all validators pay auction fee - // the clearing price in the fibonacci sequence is is 1 - let (alice_bid, _) = proof_of_fee::current_bid(*vector::borrow(&vals, 0)); - assert!(alice_bid == 1, 03); - (vals, bids, expiry) - } - - #[test_only] - public fun mock_bids(vals: &vector
): (vector, vector) { - // system_addresses::assert_ol(vm); - let bids = vector::empty(); - let expiry = vector::empty(); - let i = 0; - let prev = 0; - let fib = 1; - while (i < vector::length(vals)) { - - vector::push_back(&mut expiry, 1000); - let b = prev + fib; - vector::push_back(&mut bids, b); - - let a = vector::borrow(vals, i); - let sig = account::create_signer_for_test(*a); - // initialize and set. - proof_of_fee::pof_update_bid(&sig, b, 1000); - prev = fib; - fib = b; - i = i + 1; - }; - - (bids, expiry) - - } - - use diem_framework::chain_status; - #[test_only] - public fun ol_test_genesis(root: &signer) { - system_addresses::assert_ol(root); - genesis::setup(); - genesis::test_end_genesis(root); - - let mint_cap = init_coin_impl(root); - libra_coin::restore_mint_cap(root, mint_cap); - - assert!(!chain_status::is_genesis(), 0); - } - - #[test_only] - public fun ol_initialize_coin(root: &signer) { - system_addresses::assert_ol(root); - - let mint_cap = init_coin_impl(root); - - libra_coin::restore_mint_cap(root, mint_cap); - } - - #[test_only] - public fun ol_mint_to(root: &signer, addr: address, amount: u64) { - system_addresses::assert_ol(root); - - - let mint_cap = if (coin::is_coin_initialized()) { - libra_coin::extract_mint_cap(root) - } else { - init_coin_impl(root) - }; + public fun mock_case_1(vm: &signer, addr: address) { + assert!(stake::is_valid(addr), 01); + stake::mock_performance(vm, addr, 100, 10); + let (compliant, _, _) = grade::get_validator_grade(addr); + assert!(compliant, 01); + } + + + #[test_only] + // did not do enough mining, but did validate. + public fun mock_case_4(vm: &signer, addr: address) { + assert!(stake::is_valid(addr), 01); + stake::mock_performance(vm, addr, 0, 100); // 100 failing proposals + // let (compliant, _, _, _) = grade::get_validator_grade(addr, 0); + let (compliant, _, _) = grade::get_validator_grade(addr); + assert!(!compliant, 02); + } + + // Mock all nodes being compliant case 1 + #[test_only] + public fun mock_all_vals_good_performance(vm: &signer) { + let vals = stake::get_current_validators(); + let i = 0; + while (i < vector::length(&vals)) { + let a = vector::borrow(&vals, i); + mock_case_1(vm, *a); + i = i + 1; + }; + } + + // Mock some nodes not compliant + /*#[test_only] + public fun mock_some_vals_bad_performance(vm: &signer, bad_vals: u64) { + let vals = stake::get_current_validators(); + let vals_compliant = vector::length(&vals) - bad_vals; + let i = 0; + while (i < vals_compliant) { + let a = vector::borrow(&vals, i); + stake::mock_performance(vm, *a, 0, 100); // 100 failing proposals + i = i + 1; + }; + }*/ + + //////// PROOF OF FEE //////// + #[test_only] + public fun pof_default(): (vector
, vector, vector){ + + let vals = stake::get_current_validators(); + + let (bids, expiry) = mock_bids(&vals); + + // make all validators pay auction fee + // the clearing price in the fibonacci sequence is is 1 + let (alice_bid, _) = proof_of_fee::current_bid(*vector::borrow(&vals, 0)); + assert!(alice_bid == 1, 03); + (vals, bids, expiry) + } + + #[test_only] + public fun mock_bids(vals: &vector
): (vector, vector) { + // system_addresses::assert_ol(vm); + let bids = vector::empty(); + let expiry = vector::empty(); + let i = 0; + let prev = 0; + let fib = 1; + while (i < vector::length(vals)) { + + vector::push_back(&mut expiry, 1000); + let b = prev + fib; + vector::push_back(&mut bids, b); + + let a = vector::borrow(vals, i); + let sig = account::create_signer_for_test(*a); + // initialize and set. + proof_of_fee::pof_update_bid(&sig, b, 1000); + prev = fib; + fib = b; + i = i + 1; + }; + + (bids, expiry) + } + + use diem_framework::chain_status; + #[test_only] + public fun ol_test_genesis(root: &signer) { + system_addresses::assert_ol(root); + genesis::setup(); + genesis::test_end_genesis(root); + + let mint_cap = init_coin_impl(root); + libra_coin::restore_mint_cap(root, mint_cap); + + assert!(!chain_status::is_genesis(), 0); + } + + #[test_only] + public fun ol_initialize_coin(root: &signer) { + system_addresses::assert_ol(root); + + let mint_cap = init_coin_impl(root); + + libra_coin::restore_mint_cap(root, mint_cap); + } + + #[test_only] + public fun ol_mint_to(root: &signer, addr: address, amount: u64) { + system_addresses::assert_ol(root); + + let mint_cap = if (coin::is_coin_initialized()) { + libra_coin::extract_mint_cap(root) + } else { + init_coin_impl(root) + }; + + let c = coin::test_mint(amount, &mint_cap); + ol_account::deposit_coins(addr, c); + + let b = coin::balance(addr); + assert!(b == amount, 0001); + + libra_coin::restore_mint_cap(root, mint_cap); + } + + #[test_only] + public fun ol_initialize_coin_and_fund_vals(root: &signer, amount: u64, + drip: bool) { + system_addresses::assert_ol(root); + + let mint_cap = if (coin::is_coin_initialized()) { + libra_coin::extract_mint_cap(root) + } else { + init_coin_impl(root) + }; + + let vals = stake::get_current_validators(); + let i = 0; + while (i < vector::length(&vals)) { + let addr = vector::borrow(&vals, i); let c = coin::test_mint(amount, &mint_cap); - ol_account::deposit_coins(addr, c); + ol_account::deposit_coins(*addr, c); - let b = coin::balance(addr); + let b = libra_coin::balance(*addr); assert!(b == amount, 0001); - libra_coin::restore_mint_cap(root, mint_cap); - } - - #[test_only] - public fun ol_initialize_coin_and_fund_vals(root: &signer, amount: u64, - drip: bool) { - system_addresses::assert_ol(root); - - let mint_cap = if (coin::is_coin_initialized()) { - libra_coin::extract_mint_cap(root) - } else { - init_coin_impl(root) - }; - - let vals = stake::get_current_validators(); - let i = 0; - - while (i < vector::length(&vals)) { - let addr = vector::borrow(&vals, i); - let c = coin::test_mint(amount, &mint_cap); - ol_account::deposit_coins(*addr, c); - - let b = libra_coin::balance(*addr); - assert!(b == amount, 0001); - - - i = i + 1; - }; - - if (drip) { - slow_wallet::slow_wallet_epoch_drip(root, amount); - }; - libra_coin::restore_mint_cap(root, mint_cap); - } - - #[test_only] - fun init_coin_impl(root: &signer): coin::MintCapability { - system_addresses::assert_ol(root); - - let (burn_cap, mint_cap) = libra_coin::initialize_for_test_without_aggregator_factory(root); - coin::destroy_burn_cap(burn_cap); - - transaction_fee::initialize_fee_collection_and_distribution(root, 0); - - let initial_fees = 1000000 * 100; // coin scaling * 100 coins - let tx_fees = coin::test_mint(initial_fees, &mint_cap); - transaction_fee::vm_pay_fee(root, @ol_framework, tx_fees); - let supply_pre = libra_coin::supply(); - assert!(supply_pre == initial_fees, ESUPPLY_MISMATCH); - libra_coin::test_set_final_supply(root, initial_fees); - - mint_cap - } - - #[test_only] - public fun personas(): vector
{ - let val_addr = vector::empty
(); - - vector::push_back(&mut val_addr, @0x1000a); - vector::push_back(&mut val_addr, @0x1000b); - vector::push_back(&mut val_addr, @0x1000c); - vector::push_back(&mut val_addr, @0x1000d); - vector::push_back(&mut val_addr, @0x1000e); - vector::push_back(&mut val_addr, @0x1000f); - vector::push_back(&mut val_addr, @0x10010); // g - vector::push_back(&mut val_addr, @0x10011); // h - vector::push_back(&mut val_addr, @0x10012); // i - vector::push_back(&mut val_addr, @0x10013); // k - val_addr - } - - #[test_only] - /// mock up to 6 validators alice..frank - public fun genesis_n_vals(root: &signer, num: u64): vector
{ - system_addresses::assert_ol(root); - let framework_sig = account::create_signer_for_test(@diem_framework); - ol_test_genesis(&framework_sig); - // need to initialize musical chairs separate from genesis. - let musical_chairs_default_seats = 10; - musical_chairs::initialize(root, musical_chairs_default_seats); - - let val_addr = personas(); - let i = 0; - while (i < num) { - let val = vector::borrow(&val_addr, i); - let sig = account::create_signer_for_test(*val); - - let (_sk, pk, pop) = stake::generate_identity(); - - validator_universe::test_register_validator(root, &pk, &pop, &sig, 100, - true, true); - - vouch::init(&sig); - vouch::test_set_buddies(*val, val_addr); - - i = i + 1; - }; - - timestamp::fast_forward_seconds(2); // or else reconfigure wont happen - stake::test_reconfigure(root, validator_universe::get_eligible_validators()); - - stake::get_current_validators() - } - - #[test_only] - const EPOCH_DURATION: u64 = 60; - - #[test_only] - // NOTE: The order of these is very important. - // ol first runs its own accounting at end of epoch with epoch_boundary - // Then the stake module needs to update the validators. - // the reconfiguration module must run last, since no other - // transactions or operations can happen after the reconfig. - public fun trigger_epoch(root: &signer) { - trigger_epoch_exactly_at( - root, - reconfiguration::get_current_epoch(), - block::get_current_block_height() - ); - } - - public fun trigger_epoch_exactly_at(root: &signer, old_epoch: u64, round: u64) { - timestamp::fast_forward_seconds(EPOCH_DURATION); - - epoch_boundary::ol_reconfigure_for_test(root, old_epoch, round); - - // always advance - assert!(reconfiguration::get_current_epoch() > old_epoch, - EDID_NOT_ADVANCE_EPOCH); - - // epoch helper should always be in sync - assert!(reconfiguration::get_current_epoch() == - epoch_helper::get_current_epoch(), 666); - } + i = i + 1; + }; + + if (drip) { + slow_wallet::slow_wallet_epoch_drip(root, amount); + }; + libra_coin::restore_mint_cap(root, mint_cap); + } + + #[test_only] + fun init_coin_impl(root: &signer): coin::MintCapability { + system_addresses::assert_ol(root); + + let (burn_cap, mint_cap) = libra_coin::initialize_for_test_without_aggregator_factory(root); + coin::destroy_burn_cap(burn_cap); + + transaction_fee::initialize_fee_collection_and_distribution(root, 0); + + let initial_fees = 5_000_000 * 100; // coin scaling * 100 coins + let tx_fees = coin::test_mint(initial_fees, &mint_cap); + transaction_fee::vm_pay_fee(root, @ol_framework, tx_fees); + + // Forge Bruce + let fortune = 100_000_000_000; + let bruce_address = @0xBA7; + ol_account::create_account(root, bruce_address); + + // Bruce mints a fortune + let bruce = account::create_signer_for_test(bruce_address); + let fortune_mint = coin::test_mint(fortune, &mint_cap); + ol_account::deposit_coins(bruce_address, fortune_mint); + + // Bruce funds infra escrow + let framework = signer::address_of(root); + pledge_accounts::user_pledge(&bruce, framework, 37_000_000_000); + + let supply_pre = libra_coin::supply(); + assert!(supply_pre == (initial_fees + fortune), ESUPPLY_MISMATCH); + libra_coin::test_set_final_supply(root, initial_fees); + + mint_cap + } + + #[test_only] + public fun personas(): vector
{ + let val_addr = vector::empty
(); + vector::push_back(&mut val_addr, @0x1000a); + vector::push_back(&mut val_addr, @0x1000b); + vector::push_back(&mut val_addr, @0x1000c); + vector::push_back(&mut val_addr, @0x1000d); + vector::push_back(&mut val_addr, @0x1000e); + vector::push_back(&mut val_addr, @0x1000f); + vector::push_back(&mut val_addr, @0x10010); // g + vector::push_back(&mut val_addr, @0x10011); // h + vector::push_back(&mut val_addr, @0x10012); // i + vector::push_back(&mut val_addr, @0x10013); // k + val_addr + } + + #[test_only] + /// mock up to 6 validators alice..frank + public fun genesis_n_vals(root: &signer, num: u64): vector
{ + system_addresses::assert_ol(root); + let framework_sig = account::create_signer_for_test(@diem_framework); + ol_test_genesis(&framework_sig); + // need to initialize musical chairs separate from genesis. + let musical_chairs_default_seats = 10; + musical_chairs::initialize(root, musical_chairs_default_seats); + + let val_addr = personas(); + let i = 0; + while (i < num) { + let val = vector::borrow(&val_addr, i); + let sig = account::create_signer_for_test(*val); + let (_sk, pk, pop) = stake::generate_identity(); + validator_universe::test_register_validator(root, &pk, &pop, &sig, 100, true, true); + + vouch::init(&sig); + vouch::test_set_buddies(*val, val_addr); + + i = i + 1; + }; + + timestamp::fast_forward_seconds(2); // or else reconfigure wont happen + stake::test_reconfigure(root, validator_universe::get_eligible_validators()); + + stake::get_current_validators() + } + + #[test_only] + const EPOCH_DURATION: u64 = 60; + + #[test_only] + // NOTE: The order of these is very important. + // ol first runs its own accounting at end of epoch with epoch_boundary + // Then the stake module needs to update the validators. + // the reconfiguration module must run last, since no other + // transactions or operations can happen after the reconfig. + public fun trigger_epoch(root: &signer) { + trigger_epoch_exactly_at( + root, + reconfiguration::get_current_epoch(), + block::get_current_block_height() + ); + } + + public fun trigger_epoch_exactly_at(root: &signer, old_epoch: u64, round: u64) { + timestamp::fast_forward_seconds(EPOCH_DURATION); + epoch_boundary::ol_reconfigure_for_test(root, old_epoch, round); + + // always advance + assert!(reconfiguration::get_current_epoch() > old_epoch, + EDID_NOT_ADVANCE_EPOCH); + + // epoch helper should always be in sync + assert!(reconfiguration::get_current_epoch() == epoch_helper::get_current_epoch(), 666); + } //////// META TESTS //////// @@ -339,11 +356,12 @@ module ol_framework::mock { let n_vals = 5; let _vals = genesis_n_vals(root, n_vals); // need to include eve to init funds - let genesis_mint = 1000000; + let genesis_mint = 1_000_000; ol_initialize_coin_and_fund_vals(root, genesis_mint, true); let supply_pre = libra_coin::supply(); - let mocked_tx_fees = 1000000 * 100; - assert!(supply_pre == mocked_tx_fees + (n_vals * genesis_mint), 73570001); + let bruce_fortune = 100_000_000_000; + let mocked_tx_fees = 5_000_000 * 100; + assert!(supply_pre == bruce_fortune + mocked_tx_fees + (n_vals * genesis_mint), 73570001); } @@ -358,9 +376,9 @@ module ol_framework::mock { let (nominal_reward, entry_fee, clearing_percent, median_bid ) = proof_of_fee::get_consensus_reward(); - assert!(nominal_reward == 1000000, 73570001); + assert!(nominal_reward == 1_000_000, 73570001); assert!(clearing_percent == 1, 73570002); - assert!(entry_fee == 999, 73570003); + assert!(entry_fee == 1_000, 73570003); assert!(median_bid == 3, 73570004); } } diff --git a/framework/libra-framework/sources/ol_sources/musical_chairs.move b/framework/libra-framework/sources/ol_sources/musical_chairs.move index fee0eff82..8e3283a8b 100644 --- a/framework/libra-framework/sources/ol_sources/musical_chairs.move +++ b/framework/libra-framework/sources/ol_sources/musical_chairs.move @@ -1,240 +1,273 @@ module ol_framework::musical_chairs { - use diem_framework::chain_status; - use diem_framework::system_addresses; - use diem_framework::stake; - use ol_framework::grade; - use ol_framework::testnet; - use std::fixed_point32; - use std::vector; - - // use diem_std::debug::print; - - friend diem_framework::genesis; - friend diem_framework::diem_governance; - friend ol_framework::epoch_boundary; - #[test_only] - friend ol_framework::mock; - - - /// we don't want to play the validator selection games + use std::fixed_point32; + use std::vector; + use diem_framework::chain_status; + use diem_framework::system_addresses; + use diem_framework::stake; + use ol_framework::grade; + use ol_framework::testnet; + //use diem_std::debug::print; + + friend diem_framework::genesis; + friend diem_framework::diem_governance; + friend ol_framework::epoch_boundary; + #[test_only] + friend ol_framework::mock; + + /// We don't want to play the validator selection games /// before we're clear out of genesis const EPOCH_TO_START_EVAL: u64 = 2; - - /// we can't evaluate the performance of validators + /// We can't evaluate the performance of validators /// when there are too few rounds committed const MINIMUM_ROUNDS_PER_EPOCH: u64 = 1000; - - struct Chairs has key { - // The number of chairs in the game - seats_offered: u64, - // TODO: A small history, for future use. - history: vector, - } - - // With musical chairs we are trying to estimate - // the number of nodes which the network can support - // BFT has upperbounds in the low hundreds, but we - // don't need to hard code it. - // There also needs to be an upper bound so that there is some - // competition among validators. - // Instead of hard coding a number, and needing to reach social - // consensus to change it: we'll determine the size based on - // the network's performance as a whole. - // There are many metrics that can be used. For now we'll use - // a simple heuristic that is already on chain: compliant node cardinality. - // Other heuristics may be explored, so long as the information - // reliably committed to the chain. - - // The rules: - // Validators who perform, are not guaranteed entry into the - // next epoch of the chain. All we are establishing is the ceiling - // for validators. - // When the 100% of the validators are performing well - // the network can safely increase the threshold by 1 node. - // We can also say if less that 5% fail, no change happens. - // When the network is performing poorly, greater than 5%, - // the threshold is reduced not by a predetermined unit, but - // to the number of compliant and performant nodes. - - /// Called by root in genesis to initialize the GAS coin - public(friend) fun initialize( - vm: &signer, - genesis_seats: u64, - ) { - // system_addresses::assert_vm(vm); - // TODO: replace with VM - system_addresses::assert_ol(vm); - - chain_status::is_genesis(); - if (exists(@ol_framework)) { - return - }; - - move_to(vm, Chairs { - seats_offered: genesis_seats, - history: vector::empty(), - }); - } - - /// get the number of seats in the game - /// returns the list of compliant validators and the number of seats - /// we should offer in the next epoch - /// (compliant_vals, seats_offered) - public(friend) fun stop_the_music( // sorry, had to. - vm: &signer, - epoch: u64, - round: u64, - ): (vector
, u64) acquires Chairs { - system_addresses::assert_ol(vm); - - let validators = stake::get_current_validators(); - - let (compliant_vals, _non, fail_ratio) = - eval_compliance_impl(validators, epoch, round); - - let chairs = borrow_global_mut(@ol_framework); - - let num_compliant_nodes = vector::length(&compliant_vals); - - - // Error handle. We should not have gone into an epoch where we - // had MORE validators than seats offered. - // If this happens it's because we are in some kind of a fork condition. - // return with no changes - if (num_compliant_nodes > chairs.seats_offered) { - return (compliant_vals, chairs.seats_offered) - }; - - // The happiest case. All filled seats performed well in the last epoch - if (fixed_point32::is_zero(*&fail_ratio)) { // handle this here to prevent multiplication error below - chairs.seats_offered = chairs.seats_offered + 1; - return (compliant_vals, chairs.seats_offered) - }; - - // Conditions under which seats should be one more than the number of compliant nodes(<= 5%) - // Sad case. If we are not getting compliance, need to ratchet down the offer of seats in next epoch. - // See below find_safe_set_size, how we determine what that number - // should be - let non_compliance_pct = fixed_point32::multiply_u64(100, *&fail_ratio); - - if (non_compliance_pct > 5) { - // If network is bootstrapping don't reduce the seat count below - // compliant nodes, - - chairs.seats_offered = num_compliant_nodes; - - } else { - // Ok case. If it's between 0 and 5% then we accept that margin as if it was fully compliant - chairs.seats_offered = chairs.seats_offered + 1; - }; - - // catch failure mode - // mostly for genesis, or testnets - if (chairs.seats_offered < 4) { - chairs.seats_offered = 4; - }; - - (compliant_vals, chairs.seats_offered) - } - - // Update seat count to match filled seats post-PoF auction. - // in case we were not able to fill all the seats offered - // we don't want to keep incrementing from a baseline which we cannot fill - // it can spiral out of range. - public(friend) fun set_current_seats(framework: &signer, filled_seats: u64): u64 acquires Chairs{ - system_addresses::assert_diem_framework(framework); - let chairs = borrow_global_mut(@ol_framework); - chairs.seats_offered = filled_seats; + struct Chairs has key { + // The number of chairs in the game + seats_offered: u64, + // TODO: A small history, for future use. + history: vector, + } + + // With musical chairs we are trying to estimate + // the number of nodes which the network can support + // BFT has upperbounds in the low hundreds, but we + // don't need to hard code it. + // There also needs to be an upper bound so that there is some + // competition among validators. + // Instead of hard coding a number, and needing to reach social + // consensus to change it: we'll determine the size based on + // the network's performance as a whole. + // There are many metrics that can be used. For now we'll use + // a simple heuristic that is already on chain: compliant node cardinality. + // Other heuristics may be explored, so long as the information + // reliably committed to the chain. + + // The rules: + // Validators who perform, are not guaranteed entry into the + // next epoch of the chain. All we are establishing is the ceiling + // for validators. + // When the 100% of the validators are performing well + // the network can safely increase the threshold by 1 node. + // We can also say if less that 5% fail, no change happens. + // When the network is performing poorly, greater than 5%, + // the threshold is reduced not by a predetermined unit, but + // to the number of compliant and performant nodes. + + /// Called by root in genesis to initialize the GAS coin + public(friend) fun initialize( + vm: &signer, + genesis_seats: u64, + ) { + // system_addresses::assert_vm(vm); + // TODO: replace with VM + system_addresses::assert_ol(vm); + + chain_status::is_genesis(); + if (exists(@ol_framework)) { + return + }; + + move_to(vm, Chairs { + seats_offered: genesis_seats, + history: vector::empty(), + }); + } + + /// get the number of seats in the game + /// returns the list of compliant validators and the number of seats + /// we should offer in the next epoch + /// (compliant_vals, seats_offered) + public(friend) fun stop_the_music( + vm: &signer, + epoch: u64, + round: u64, + ): (vector
, u64) acquires Chairs { + system_addresses::assert_ol(vm); + + let chairs = borrow_global_mut(@ol_framework); + let validators = stake::get_current_validators(); + let (compliant_vals, _non, fail_ratio) = eval_compliance_impl(validators, epoch, round); + let num_compliant_vals = vector::length(&compliant_vals); + + // Error handle. We should not have gone into an epoch where we had MORE validators than seats offered. + // If this happens it's because we are in some kind of a fork condition. Return with no changes + if (num_compliant_vals > chairs.seats_offered) { + return (compliant_vals, chairs.seats_offered) + }; + + // Calculate the number of seats to offer + chairs.seats_offered = calculate_seats_to_offer( + num_compliant_vals, + fail_ratio, chairs.seats_offered + ); + + (compliant_vals, chairs.seats_offered) + } + + // Case: All seats performed well in the last epoch + // - The happiest case: All filled seats performed well in the last epoch. + // - Increment the number of seats offered by 1. + // + // Case: Non-compliance detected + // - If non-compliance is detected (> 5%), reduce the number of seats offered to the number of compliant nodes. + // - Otherwise, if non-compliance is between 0 and 5%, increment the number of seats offered by 1. + // + // Error Handling + // - If there are more validators than seats offered, it indicates a fork condition. In such a case, return with no changes. + // + // Bootstrap Mode + // - If the network is bootstrapping, ensure that the seat count does not fall below the number of compliant nodes. + // - Ensure that the number of seats offered is not less than 4, especially during genesis or in testnets. + + public fun calculate_seats_to_offer( + num_compliant_vals: u64, + fail_ratio: fixed_point32::FixedPoint32, + current_seats_offered: u64 + ): u64 { + let new_seats_offered = if (fixed_point32::is_zero(fail_ratio)) { + // The happiest case. All filled seats performed well in the last epoch + current_seats_offered + 1 + } else { + let non_compliance_pct = fixed_point32::multiply_u64(100, fail_ratio); + if (non_compliance_pct > 5) { + // Sad case. If we are not getting compliance, need to ratchet down the offer of seats in the next epoch. + // See below find_safe_set_size, how we determine what that number should be + num_compliant_vals + } else { + // Ok case. If it's between 0 and 5% then we accept that margin as if it was fully compliant + current_seats_offered + 1 + } + }; + + // Catch failure mode mostly for genesis, or testnets + if (new_seats_offered < 4) { + 4 + } else { + new_seats_offered } - - // use the Case statistic to determine what proportion of the network is compliant. - // private function prevent list DoS. - fun eval_compliance_impl( - validators: vector
, - epoch: u64, - _round: u64, - ): (vector
, vector
, fixed_point32::FixedPoint32) { - - let val_set_len = vector::length(&validators); - - // skip first epoch, don't evaluate if the network doesn't have sufficient data - if (epoch < EPOCH_TO_START_EVAL) return (validators, vector::empty
(), fixed_point32::create_from_rational(1, 1)); - - let (compliant_nodes, non_compliant_nodes) = grade::eval_compliant_vals(validators); - - let good_len = vector::length(&compliant_nodes) ; - let bad_len = vector::length(&non_compliant_nodes); - - // Note: sorry for repetition but necessary for writing tests and debugging. - let null = fixed_point32::create_from_raw_value(0); - if (good_len > val_set_len) { // safety - return (vector::empty(), vector::empty(), null) - }; - - if (bad_len > val_set_len) { // safety - return (vector::empty(), vector::empty(), null) - }; - - if ((good_len + bad_len) != val_set_len) { // safety - return (vector::empty(), vector::empty(), null) - }; - - let ratio = if (bad_len > 0) { - fixed_point32::create_from_rational(bad_len, val_set_len) - } else { - null - }; - - (compliant_nodes, non_compliant_nodes, ratio) - } - - // Check for genesis, upgrade or recovery mode scenarios - // if we are at genesis or otherwise at start of an epoch and don't - // have a sufficient amount of history to evaluate nodes - // we might reduce the validator set too agressively. - // Musical chairs should not evaluate performance with less than 1000 rounds - // created on mainnet, - // there's something else very wrong in that case. - - fun is_booting_up(epoch: u64, round: u64): bool { - !testnet::is_testnet() && - (epoch < EPOCH_TO_START_EVAL || - round < MINIMUM_ROUNDS_PER_EPOCH) - } - - //////// GETTERS //////// - - #[view] - public fun get_current_seats(): u64 acquires Chairs { - borrow_global(@ol_framework).seats_offered - } - - //////// TEST HELPERS //////// - - #[test_only] - use diem_framework::chain_id; - - #[test_only] - public fun test_stop(vm: &signer, epoch: u64, epoch_round: u64): (vector
, u64) acquires Chairs { - stop_the_music(vm, epoch, epoch_round) - } - - #[test_only] - public fun test_eval_compliance(root: &signer, validators: vector
, - epoch: u64, round: u64): (vector
, vector
, - fixed_point32::FixedPoint32) { - system_addresses::assert_ol(root); - eval_compliance_impl(validators, epoch, round) - - } - //////// TESTS //////// - - #[test(vm = @ol_framework)] - public entry fun initialize_chairs(vm: signer) acquires Chairs { - chain_id::initialize_for_test(&vm, 4); - initialize(&vm, 10); - assert!(get_current_seats() == 10, 1004); - } + } + + // Update seat count to match filled seats post-PoF auction. + // in case we were not able to fill all the seats offered + // we don't want to keep incrementing from a baseline which we cannot fill + // it can spiral out of range. + public(friend) fun set_current_seats(framework: &signer, filled_seats: u64): u64 acquires Chairs{ + system_addresses::assert_diem_framework(framework); + let chairs = borrow_global_mut(@ol_framework); + chairs.seats_offered = filled_seats; + chairs.seats_offered + } + + // use the Case statistic to determine what proportion of the network is compliant. + // private function prevent list DoS. + fun eval_compliance_impl( + validators: vector
, + epoch: u64, + _round: u64, + ): (vector
, vector
, fixed_point32::FixedPoint32) { + let val_set_len = vector::length(&validators); + + // skip first epoch, don't evaluate if the network doesn't have sufficient data + if (epoch < EPOCH_TO_START_EVAL) return (validators, vector::empty
(), fixed_point32::create_from_rational(1, 1)); + + let (compliant_nodes, non_compliant_nodes) = grade::eval_compliant_vals(validators); + + let good_len = vector::length(&compliant_nodes) ; + let bad_len = vector::length(&non_compliant_nodes); + + // Note: sorry for repetition but necessary for writing tests and debugging. + let null = fixed_point32::create_from_raw_value(0); + if (good_len > val_set_len) { // safety + return (vector::empty(), vector::empty(), null) + }; + + if (bad_len > val_set_len) { // safety + return (vector::empty(), vector::empty(), null) + }; + + if ((good_len + bad_len) != val_set_len) { // safety + return (vector::empty(), vector::empty(), null) + }; + + let ratio = if (bad_len > 0) { + fixed_point32::create_from_rational(bad_len, val_set_len) + } else { + null + }; + + (compliant_nodes, non_compliant_nodes, ratio) + } + + // Check for genesis, upgrade or recovery mode scenarios + // if we are at genesis or otherwise at start of an epoch and don't + // have a sufficient amount of history to evaluate nodes + // we might reduce the validator set too agressively. + // Musical chairs should not evaluate performance with less than 1000 rounds + // created on mainnet, + // there's something else very wrong in that case. + + fun is_booting_up(epoch: u64, round: u64): bool { + !testnet::is_testnet() && + (epoch < EPOCH_TO_START_EVAL || + round < MINIMUM_ROUNDS_PER_EPOCH) + } + + //////// GETTERS //////// + + #[view] + public fun get_current_seats(): u64 acquires Chairs { + borrow_global(@ol_framework).seats_offered + } + + //////// TEST HELPERS //////// + + #[test_only] + use diem_framework::chain_id; + + #[test_only] + public fun test_stop(vm: &signer, epoch: u64, epoch_round: u64): (vector
, u64) acquires Chairs { + stop_the_music(vm, epoch, epoch_round) + } + + #[test_only] + public fun test_eval_compliance(root: &signer, validators: vector
, + epoch: u64, round: u64): (vector
, vector
, + fixed_point32::FixedPoint32) { + system_addresses::assert_ol(root); + eval_compliance_impl(validators, epoch, round) + + } + + //////// TESTS //////// + + #[test(vm = @ol_framework)] + public entry fun initialize_chairs(vm: signer) acquires Chairs { + chain_id::initialize_for_test(&vm, 4); + initialize(&vm, 10); + assert!(get_current_seats() == 10, 1004); + } + + #[test] + fun test_calculate_seats_to_offer() { + // Test case: all seats are compliant + let result = calculate_seats_to_offer(5, fixed_point32::create_from_rational(0, 1), 5); + assert!(result == 6, 1001); + + // Test case: more than 5% non-compliance + let result = calculate_seats_to_offer(18, fixed_point32::create_from_rational(2, 20), 20); + assert!(result == 18, 1002); + + // Test case: 5% non-compliance + let result = calculate_seats_to_offer(19, fixed_point32::create_from_rational(1, 20), 20); + assert!(result == 21, 1003); + + // Test case: less than 5% non-compliance + let result = calculate_seats_to_offer(98, fixed_point32::create_from_rational(2, 100), 100); + assert!(result == 101, 1003); + + // Test case: seats offered should not be less than 4 + let result = calculate_seats_to_offer(2, fixed_point32::create_from_rational(1, 1), 2); + assert!(result == 4, 1004); + } } diff --git a/framework/libra-framework/sources/ol_sources/pledge_accounts.move b/framework/libra-framework/sources/ol_sources/pledge_accounts.move index 1aebfd115..c25cf387c 100644 --- a/framework/libra-framework/sources/ol_sources/pledge_accounts.move +++ b/framework/libra-framework/sources/ol_sources/pledge_accounts.move @@ -51,12 +51,15 @@ use diem_framework::coin; use diem_framework::system_addresses; - // use diem_std::debug::print; + //use diem_std::debug::print; friend ol_framework::infra_escrow; friend ol_framework::genesis_migration; friend ol_framework::genesis; + #[test_only] + friend ol_framework::mock; + /// no policy at this address const ENO_BENEFICIARY_POLICY: u64 = 1; /// there is a non zero balance @@ -160,7 +163,7 @@ sig: &signer, address_of_beneficiary: address, pledge: coin::Coin - ) acquires MyPledges, BeneficiaryPolicy { + ) acquires MyPledges, BeneficiaryPolicy { maybe_initialize_my_pledges(sig); assert!(exists(address_of_beneficiary), error::invalid_state(ENO_BENEFICIARY_POLICY)); let sender_addr = signer::address_of(sig); @@ -248,57 +251,56 @@ // withdraw an amount from all pledge accounts. Check first that there are remaining funds before attempting to withdraw. public(friend) fun withdraw_from_all_pledge_accounts(sig_beneficiary: &signer, amount: u64): option::Option> acquires MyPledges, BeneficiaryPolicy { - let address_of_beneficiary = signer::address_of(sig_beneficiary); - if (!exists(address_of_beneficiary)) { - return option::none>() - }; - - let pledgers = *&borrow_global(address_of_beneficiary).pledgers; - let amount_available = *&borrow_global(address_of_beneficiary).amount_available; - - if (amount_available == 0 || amount == 0) { - return option::none>() - }; - - let pct_withdraw = fixed_point64::create_from_rational((amount as u128), (amount_available as u128)); - - let i = 0; - let all_coins = option::none>(); - while (i < vector::length(&pledgers)) { - let pledge_account = *vector::borrow(&pledgers, i); - - // DANGER: this is a private function that changes balances. - if (!exists(pledge_account)) continue; + let address_of_beneficiary = signer::address_of(sig_beneficiary); + if (!exists(address_of_beneficiary)) { + return option::none>() + }; - let c = withdraw_pct_from_one_pledge_account(&address_of_beneficiary, &pledge_account, &pct_withdraw); - // GROSS: dealing with options in Move. - // TODO: find a better way. - if (option::is_none(&all_coins) && option::is_some(&c)) { + let pledgers = *&borrow_global(address_of_beneficiary).pledgers; + let amount_available = *&borrow_global(address_of_beneficiary).amount_available; - let coin = option::extract(&mut c); - option::fill(&mut all_coins, coin); - option::destroy_none(c); - } else if (option::is_some(&c)) { + if (amount_available == 0 || amount == 0) { + return option::none>() + }; - let temp = option::extract(&mut all_coins); - let coin = option::extract(&mut c); - libra_coin::merge(&mut temp, coin); - option::destroy_none(all_coins); - all_coins = option::some(temp); - option::destroy_none(c); - } else { - option::destroy_none(c); - }; + let pct_withdraw = fixed_point64::create_from_rational((amount as u128), (amount_available as u128)); - i = i + 1; + let i = 0; + let all_coins = option::none>(); + while (i < vector::length(&pledgers)) { + let pledge_account = *vector::borrow(&pledgers, i); + + // DANGER: this is a private function that changes balances. + if (!exists(pledge_account)) continue; + + let c = withdraw_pct_from_one_pledge_account(&address_of_beneficiary, &pledge_account, &pct_withdraw); + // GROSS: dealing with options in Move. + // TODO: find a better way. + if (option::is_none(&all_coins) && option::is_some(&c)) { + + let coin = option::extract(&mut c); + option::fill(&mut all_coins, coin); + option::destroy_none(c); + } else if (option::is_some(&c)) { + + let temp = option::extract(&mut all_coins); + let coin = option::extract(&mut c); + libra_coin::merge(&mut temp, coin); + option::destroy_none(all_coins); + all_coins = option::some(temp); + option::destroy_none(c); + } else { + option::destroy_none(c); }; + i = i + 1; + }; + all_coins } - fun get_user_pledges(account: &address): vector
acquires - MyPledges { + fun get_user_pledges(account: &address): vector
acquires MyPledges { let list = vector::empty
(); if (!exists(*account)) return list; let user_state = borrow_global(*account); diff --git a/framework/libra-framework/sources/ol_sources/proof_of_fee.move b/framework/libra-framework/sources/ol_sources/proof_of_fee.move index 1dcb3e87a..16ff5ff2d 100644 --- a/framework/libra-framework/sources/ol_sources/proof_of_fee.move +++ b/framework/libra-framework/sources/ol_sources/proof_of_fee.move @@ -4,26 +4,26 @@ ///////////////////////////////////////////////////////////////////////// // NOTE: this module replaces NodeWeight.move, which becomes redundant since // all validators have equal weight in consensus. -// TODO: the bubble sort functions were lifted directly from NodeWeight, needs checking. /////////////////////////////////////////////////////////////////////////// - module ol_framework::proof_of_fee { use std::error; use std::signer; use std::vector; + use std::math64; use std::fixed_point32; use diem_framework::validator_universe; - use ol_framework::jail; - use ol_framework::slow_wallet; - use ol_framework::vouch; use diem_framework::transaction_fee; - use ol_framework::epoch_helper; use diem_framework::account; use diem_framework::stake; use diem_framework::system_addresses; + use ol_framework::jail; + use ol_framework::slow_wallet; + use ol_framework::vouch; + use ol_framework::epoch_helper; use ol_framework::globals; - // use diem_std::debug::print; + use ol_framework::address_utils; + //use diem_std::debug::print; friend diem_framework::genesis; friend ol_framework::epoch_boundary; @@ -32,14 +32,25 @@ module ol_framework::proof_of_fee { #[test_only] friend ol_framework::mock; - /// The nominal reward for each validator in each epoch. const GENESIS_BASELINE_REWARD: u64 = 1000000; /// Number of vals needed before PoF becomes competitive for /// performant nodes as well - const VAL_BOOT_UP_THRESHOLD: u64 = 19; - + const VAL_BOOT_UP_THRESHOLD: u64 = 21; + /// This figure is experimental and a different percentage may be finalized + /// after some experience in the wild. Additionally it could be dynamic + /// based on another function or simply randomized within a range + /// (as originally proposed in this feature request) + const PCT_REDUCTION_FOR_COMPETITION: u64 = 20; // 20% + /// Upper bound threshold for bid percentages. + const BID_UPPER_BOUND: u64 = 0950; // 95% + /// Lower bound threshold for bid percentages. + const BID_LOWER_BOUND: u64 = 0500; // 50% + /// Short window period for recent bid trends. + const SHORT_WINDOW: u64 = 5; // 5 epochs + /// Long window period for extended bid trends. + const LONG_WINDOW: u64 = 10; // 10 epochs //////// ERRORS ///////// /// Not an active validator @@ -63,7 +74,6 @@ module ol_framework::proof_of_fee { /// not enough coin balance const ELOW_UNLOCKED_COIN_BALANCE: u64 = 17; - // A struct on the validators account which indicates their // latest bid (and epoch) struct ProofOfFeeAuction has key { @@ -132,55 +142,84 @@ module ol_framework::proof_of_fee { } } - /// Consolidates all the logic for the epoch boundary/ - /// Includes: getting the sorted bids, - /// filling the seats (determined by MusicalChairs), and getting a price. - /// and finally charging the validators for their bid (everyone pays the lowest) - /// for audit instrumentation returns: final set size, auction winners, all the bidders, (including not-qualified), and all qualified bidders. - /// we also return the auction entry price (clearing price) + /// Consolidates all the logic for the epoch boundary, including: + /// 1. Getting the sorted bidders, + /// 2. Calculate final validators set size (number of seats to fill), + /// 3. Filling the seats, + /// 4. Getting a price, + /// 5. Finally charging the validators for their bid (everyone pays the lowest) + /// For audit instrumentation returns: final set size, auction winners, all the bidders, (including not-qualified), and all qualified bidders. + /// We also return the auction entry price (clearing price) /// (final_set_size, auction_winners, all_bidders, only_qualified_bidders, actually_paid, entry_fee) public(friend) fun end_epoch( vm: &signer, outgoing_compliant_set: &vector
, - final_set_size: u64 + mc_set_size: u64 // musical chairs set size suggestion ): (vector
, vector
, vector
, u64) acquires ProofOfFeeAuction, ConsensusReward { - system_addresses::assert_ol(vm); - - let all_bidders = get_bidders(false); - let only_qualified_bidders = get_bidders(true); - // The set size as determined by musical chairs is a target size - // but the actual final size depends on how much can we expand the set - // without adding too many unproven nodes (which we don't know if they are prepared to validate, and risk halting th network). - - - // Boot up - // After an upgrade or incident the network may need to rebuild the - // validator set from a small base. - // we should increase the available seats starting from a base of - // compliant nodes. And make it competitive for the unknown nodes. - // Instead of increasing the seats by +1 the compliant vals we should - // increase by compliant + (1/2 compliant - 1) or another - // safe threshold. - // Another effect is that with PoF we might be dropping compliant nodes, - // in favor of unknown nodes with high bids. - // So in the case of a small validator set, we ignore the musical_chairs - // suggestion, and increase the seats offered, and guarantee seats to - // performant nodes. - let performant_len = vector::length(outgoing_compliant_set); - if ( - performant_len < VAL_BOOT_UP_THRESHOLD && - performant_len > 2 - ) { - final_set_size = performant_len + (performant_len/2 - 1) - }; + system_addresses::assert_ol(vm); + + let all_bidders = get_bidders(false); + let only_qualified_bidders = get_bidders(true); + + // Calculate the final set size considering the number of compliant validators, + // number of qualified bidders, and musical chairs set size suggestion + let final_set_size = calculate_final_set_size( + vector::length(outgoing_compliant_set), + vector::length(&only_qualified_bidders), + mc_set_size); + + // This is the core of the mechanism, the uniform price auction + // the winners of the auction will be the validator set. + // Other lists are created for audit purposes of the BoundaryStatus + let (auction_winners, entry_fee, _clearing_bid, _proven, _unproven) = fill_seats_and_get_price(vm, final_set_size, &only_qualified_bidders, outgoing_compliant_set); + + (auction_winners, all_bidders, only_qualified_bidders, entry_fee) + } + - // This is the core of the mechanism, the uniform price auction - // the winners of the auction will be the validator set. - // other lists are created for audit purposes of the BoundaryStatus - let (auction_winners, entry_fee, _clearing_bid, _proven, _unproven) = fill_seats_and_get_price(vm, final_set_size, &only_qualified_bidders, outgoing_compliant_set); + // The set size as determined by musical chairs is a target size + // but the actual final size depends on: + // a. how much can we expand the set without adding too many unproven nodes (which we don't know + // if they are prepared to validate, and risk halting the network). + // b. how many qualified bidders are there, and how many seats can we offer to ensure competition. + + // 1. Boot Up + // After an upgrade or incident the network may need to rebuild the validator set + // from a small base. We should increase the available seats starting + // from a base of compliant nodes. And make it competitive for the unknown nodes. + // Instead of increasing the seats by +1 the compliant vals we should + // increase by compliant + (1/2 compliant - 1) or another safe threshold. + // Another effect is that with PoF we might be dropping compliant nodes, + // in favor of unknown nodes with high bids. So in the case of a small validator set, + // we ignore the musical_chairs suggestion, and increase the seats offered, and guarantee seats to + // performant nodes. + // + // 2. Competitive Set + // If we have more qualified bidders than the threshold, we should limit the final set size + // to 90% of the qualified bidders to ensure that vals will compete for seats. + + fun calculate_final_set_size( + outgoing_compliant: u64, + qualified_bidders: u64, + mc_set_size: u64 // musical chairs set size suggestion + ): u64 { + // 1. Boot Up + if ( + outgoing_compliant < VAL_BOOT_UP_THRESHOLD && + outgoing_compliant > 2 + ) { + return math64::min(outgoing_compliant + (outgoing_compliant/2 - 1), VAL_BOOT_UP_THRESHOLD) + }; + // 2. Competitive Set + if (mc_set_size >= VAL_BOOT_UP_THRESHOLD && qualified_bidders >= VAL_BOOT_UP_THRESHOLD) { + let seats_to_remove = qualified_bidders * PCT_REDUCTION_FOR_COMPETITION / 100; + let max_qualified = qualified_bidders - seats_to_remove; + // do note increase beyond musical chairs suggestion and competitive set size + return math64::min(max_qualified, mc_set_size) + }; - (auction_winners, all_bidders, only_qualified_bidders, entry_fee) + mc_set_size } /// The fees are charged seperate from the auction and seating loop @@ -197,9 +236,6 @@ module ol_framework::proof_of_fee { } - - - //////// CONSENSUS CRITICAL //////// // Get the validator universe sorted by bid // By default this will return a FILTERED list of validators @@ -234,11 +270,8 @@ module ol_framework::proof_of_fee { let k = 0; while (k < length) { // TODO: Ensure that this address is an active validator - let cur_address = *vector::borrow
(eligible_validators, k); - let (bid, _expire) = current_bid(cur_address); - let (_, qualified) = audit_qualification(cur_address); if (remove_unqualified && !qualified) { k = k + 1; @@ -249,37 +282,16 @@ module ol_framework::proof_of_fee { k = k + 1; }; - // Sorting the accounts vector based on value (weights). - // Bubble sort algorithm - let len_filtered = vector::length
(&filtered_vals); - // if there's only one person (testing) - if (len_filtered < 2) return (filtered_vals, bids); - let i = 0; - while (i < len_filtered){ - let j = 0; - while(j < len_filtered-i-1){ - - let value_j = *(vector::borrow(&bids, j)); - - let value_jp1 = *(vector::borrow(&bids, j+1)); - if(value_j > value_jp1){ - - vector::swap(&mut bids, j, j+1); - - vector::swap
(&mut filtered_vals, j, j+1); - }; - j = j + 1; - - }; - i = i + 1; - - }; - + // Sorting the accounts vector based on their bids + address_utils::sort_by_values(&mut filtered_vals, &mut bids); // Reverse to have sorted order - high to low. vector::reverse(&mut filtered_vals); vector::reverse(&mut bids); + // Shuffle duplicates to garantee randomness/fairness + address_utils::shuffle_duplicates(&mut filtered_vals, &mut bids); + return (filtered_vals, bids) } @@ -300,7 +312,7 @@ module ol_framework::proof_of_fee { // Here we place the bidders into their seats. // The order of the bids will determine placement. // One important aspect of picking the next validator set: - // it should have 2/3rds of known good ("proven") validators + // It should have 2/3rds of known good ("proven") validators // from the previous epoch. Otherwise the unproven nodes, who // may not be ready for consensus, might be offline and cause a halt. // Validators can be inattentive and have bids that qualify, but their nodes @@ -314,7 +326,9 @@ module ol_framework::proof_of_fee { // This is a potential issue again with inattentive validators who // have have a high bid, but again they fail repeatedly to finalize an epoch // successfully. Their bids should not penalize validators who don't have - // a streak of jailed epochs. So of the 1/3 unproven nodes, we'll first seat the validators with Jail.consecutive_failure_to_rejoin < 2, and after that the remainder. + // a streak of jailed epochs. So of the 1/3 unproven nodes, + // we'll first seat the validators with Jail.consecutive_failure_to_rejoin < 2, + // and after that the remainder. // There's some code implemented which is not enabled in the current form. // Unsealed auctions are tricky. The Proof Of Fee @@ -324,14 +338,18 @@ module ol_framework::proof_of_fee { // As such per epoch the validator is only allowed to revise their bids / // down once. To do this in practice they need to retract a bid (sit out // the auction), and then place a new bid. - // A validator can always leave the auction, but if they rejoin a second time in the epoch, then they've committed a bid until the next epoch. - // So retracting should be done with care. The ergonomics are not great. + // A validator can always leave the auction, but if they rejoin a second time in the epoch, + // then they've committed a bid until the next epoch. + // So retracting should be done with care. The ergonomics are not great. // The preference would be not to have this constraint if on the margins // the ergonomics brings more bidders than attackers. // After more experience in the wild, the network may decide to // limit bid retracting. - // The Validator must qualify on a number of metrics: have funds in their Unlocked account to cover bid, have miniumum viable vouches, and not have been jailed in the previous round. + // The Validator must qualify on a number of metrics: + // 1. have funds in their Unlocked account to cover bid, + // 2. have miniumum viable vouches, + // 3. and not have been jailed in the previous round. /// Showtime. /// This is where we take all the bidders and seat them. @@ -339,11 +357,11 @@ module ol_framework::proof_of_fee { /// This function assumes we have already filtered out ineligible validators. /// but we will check again here. /// we return: - // a. the list of winning validators (the validator set) - // b. the entry fee paid - // c. the clearing bid (percentage paid) - // d. the list of proven nodes added, for audit and instrumentation - // e. the list of unproven, for audit and instrumentation + /// a. the list of winning validators (the validator set) + /// b. the entry fee paid + /// c. the clearing bid (percentage paid) + /// d. the list of proven nodes added, for audit and instrumentation + /// e. the list of unproven, for audit and instrumentation public(friend) fun fill_seats_and_get_price( vm: &signer, final_set_size: u64, @@ -376,12 +394,12 @@ module ol_framework::proof_of_fee { (i < vector::length(&sorted_vals_by_bid)) ) { let val = vector::borrow(&sorted_vals_by_bid, i); - if (!account::exists_at(*val)) { - i = i + 1; - continue - }; + if (!account::exists_at(*val)) { + i = i + 1; + continue + }; // check if a proven node - // NOTE: if the top bidders all all "proven" nodes, then there will + // NOTE: if the top bidders are all "proven" nodes, then there will // be no reason to add an unproven. Unproven nodes will only // be picked if they have bids higher than the bottom 1/3 bids of the proven nodes if (vector::contains(proven_nodes, val)) { @@ -417,7 +435,7 @@ module ol_framework::proof_of_fee { cr.clearing_bid = lowest_bid_pct; if (lowest_bid_pct > 0) { - cr.entry_fee = fixed_point32::multiply_u64(cr.nominal_reward, bid_as_fixedpoint(lowest_bid_pct)); + cr.entry_fee = cr.nominal_reward * lowest_bid_pct / 1000; if (cr.nominal_reward > cr.entry_fee) { cr.net_reward = cr.nominal_reward - cr.entry_fee; @@ -472,119 +490,88 @@ module ol_framework::proof_of_fee { fun bid_as_fixedpoint(bid_pct: u64): fixed_point32::FixedPoint32 { fixed_point32::create_from_rational(bid_pct, 1000) } - /// Adjust the reward at the end of the epoch - /// as described in the paper, the epoch reward needs to be adjustable - /// given that the implicit bond needs to be sufficient, eg 5-10x the reward. + + + /// Calculates the reward adjustment based on bid history and nominal reward. + /// @param median_history - The median history of bids. + /// @param nominal_reward - The current nominal reward. /// @return Tuple (bool, bool, u64) /// 0: did the thermostat run, /// 1: did it increment, or decrease, bool /// 2: how much - /// if the thermostat returns (false, false, 0), it means there was an error running - public(friend) fun reward_thermostat(vm: &signer): (bool, bool, u64) acquires ConsensusReward { - system_addresses::assert_ol(vm); - // check the bid history - // if there are 5 days above 95% adjust the reward up by 5% - // adjust by more if it has been 10 days then, 10% - // if there are 5 days below 50% adjust the reward down. - // adjust by more if it has been 10 days then 10% - - let bid_upper_bound = 0950; - let bid_lower_bound = 0500; - - let short_window: u64 = 5; - let long_window: u64 = 10; - - let cr = borrow_global_mut(@ol_framework); - - - let len = vector::length(&cr.median_history); - let i = 0; - + fun calculate_reward_adjustment( + median_history: &vector, + nominal_reward: u64 + ): (bool, bool, u64) { + let history_length = vector::length(median_history); + let index = 0; let epochs_above = 0; let epochs_below = 0; - while (i < 16 && i < len) { // max ten days, but may have less in history, filling set should truncate the history at 15 epochs. - let avg_bid = *vector::borrow(&cr.median_history, i); + while (index < 16 && index < history_length) { + let avg_bid = *vector::borrow(median_history, index); - if (avg_bid > bid_upper_bound) { + if (avg_bid > BID_UPPER_BOUND) { epochs_above = epochs_above + 1; - } else if (avg_bid < bid_lower_bound) { + } else if (avg_bid < BID_LOWER_BOUND) { epochs_below = epochs_below + 1; }; - i = i + 1; + index = index + 1; }; - - if (cr.nominal_reward > 0) { - - - // TODO: this is an initial implementation, we need to - // decide if we want more granularity in the reward adjustment - // Note: making this readable for now, but we can optimize later + if (nominal_reward > 0) { if (epochs_above > epochs_below) { - - // if (epochs_above > short_window) { - - // check for zeros. - // TODO: put a better safety check here - - // If the Validators are bidding near 100% that means - // the reward is very generous, i.e. their opportunity - // cost is met at small percentages. This means the - // implicit bond is very high on validators. E.g. - // at 1% median bid, the implicit bond is 100x the reward. - // We need to DECREASE the reward - - - if (epochs_above > long_window) { - - // decrease the reward by 10% - let less_ten_pct = (cr.nominal_reward / 10); - cr.nominal_reward = cr.nominal_reward - less_ten_pct; + if (epochs_above > LONG_WINDOW) { + let less_ten_pct = (nominal_reward / 10); return (true, false, less_ten_pct) - - } else if (epochs_above > short_window) { - // decrease the reward by 5% - let less_five_pct = (cr.nominal_reward / 20); - cr.nominal_reward = cr.nominal_reward - less_five_pct; - + } else if (epochs_above > SHORT_WINDOW) { + let less_five_pct = (nominal_reward / 20); return (true, false, less_five_pct) } + } else { + if (epochs_below > LONG_WINDOW) { + let increase_ten_pct = (nominal_reward / 10); + return (true, true, increase_ten_pct) + } else if (epochs_below > SHORT_WINDOW) { + let increase_five_pct = (nominal_reward / 20); + return (true, true, increase_five_pct) + } }; - // return early since we can't increase and decrease simultaneously - - - // if validators are bidding low percentages - // it means the nominal reward is not high enough. - // That is the validator's opportunity cost is not met within a - // range where the bond is meaningful. - // For example: if the bids for the epoch's reward is 50% of the value, that means the potential profit, is the same as the potential loss. - // At a 25% bid (potential loss), the profit is thus 75% of the value, which means the implicit bond is 25/75, or 1/3 of the bond, the risk favors the validator. This means among other things, that an attacker can pay for the cost of the attack with the profits. See paper, for more details. + return (true, false, 0) + }; - // we need to INCREASE the reward, so that the bond is more meaningful. + (false, false, 0) + } - if (epochs_below > long_window) { - // increase the reward by 10% - let increase_ten_pct = (cr.nominal_reward / 10); - cr.nominal_reward = cr.nominal_reward + increase_ten_pct; - return (true, true, increase_ten_pct) - } else if (epochs_below > short_window) { - + /// Adjust the reward at the end of the epoch + /// as described in the paper, the epoch reward needs to be adjustable + /// given that the implicit bond needs to be sufficient, eg 5-10x the reward. + /// @param vm - The signer. + /// @return Tuple (bool, bool, u64) + /// 0: did the thermostat run, + /// 1: did it increment, or decrease, bool + /// 2: how much + /// if the thermostat returns (false, false, 0), it means there was an error running + public(friend) fun reward_thermostat(vm: &signer): (bool, bool, u64) acquires ConsensusReward { + system_addresses::assert_ol(vm); + let cr = borrow_global_mut(@ol_framework); - // increase the reward by 5% - let increase_five_pct = (cr.nominal_reward / 20); - cr.nominal_reward = cr.nominal_reward + increase_five_pct; - return (true, true, increase_five_pct) - }; + let (did_run, did_increment, amount) = calculate_reward_adjustment( + &cr.median_history, + cr.nominal_reward + ); - // we ran the thermostat but no change was made. - return (true, false, 0) + if (did_run) { + if (did_increment) { + cr.nominal_reward = cr.nominal_reward + amount; + } else { + cr.nominal_reward = cr.nominal_reward - amount; + } }; - // nominal reward is zero, there's a problem - return (false, false, 0) + (did_run, did_increment, amount) } /// find the median bid to push to history @@ -623,8 +610,6 @@ module ol_framework::proof_of_fee { } //////////////// GETTERS //////////////// - // get the current bid for a validator - #[view] /// get the baseline reward from ConsensusReward @@ -634,6 +619,7 @@ module ol_framework::proof_of_fee { return (b.nominal_reward, b.entry_fee, b.clearing_bid, b.median_win_bid) } + // get the current bid for a validator // CONSENSUS CRITICAL // ALL EYES ON THIS // Proof of Fee returns the current bid of the validator during the auction for upcoming epoch seats. @@ -665,21 +651,41 @@ module ol_framework::proof_of_fee { return (false, 0) } + #[view] + /// Query the reward adjustment without altering the nominal reward. + /// @param vm - The signer. + /// @return Tuple (bool, bool, u64) + /// 0: did the thermostat run, + /// 1: did it increment, or decrease, bool + /// 2: how much + /// if the thermostat returns (false, false, 0), it means there was an error running + public fun query_reward_adjustment(): (bool, bool, u64) acquires ConsensusReward { + let cr = borrow_global(@ol_framework); + + let (did_run, did_increment, amount) = calculate_reward_adjustment( + &cr.median_history, + cr.nominal_reward + ); + + (did_run, did_increment, amount) + } + + // Get the top N validators by bid, this is FILTERED by default public(friend) fun top_n_accounts(account: &signer, n: u64, unfiltered: bool): vector
acquires ProofOfFeeAuction, ConsensusReward { - system_addresses::assert_vm(account); + system_addresses::assert_vm(account); - let eligible_validators = get_bidders(unfiltered); - let len = vector::length
(&eligible_validators); - if(len <= n) return eligible_validators; + let eligible_validators = get_bidders(unfiltered); + let len = vector::length
(&eligible_validators); + if(len <= n) return eligible_validators; - let diff = len - n; - while(diff > 0){ - vector::pop_back(&mut eligible_validators); - diff = diff - 1; - }; + let diff = len - n; + while(diff > 0){ + vector::pop_back(&mut eligible_validators); + diff = diff - 1; + }; - eligible_validators + eligible_validators } @@ -708,13 +714,11 @@ module ol_framework::proof_of_fee { /// Note that the validator will not be bidding on any future /// epochs if they retract their bid. The must set a new bid. fun retract_bid(account_sig: &signer) acquires ProofOfFeeAuction { - let acc = signer::address_of(account_sig); if (!exists(acc)) { init(account_sig); }; - let pof = borrow_global_mut(acc); let this_epoch = epoch_helper::get_current_epoch(); @@ -725,7 +729,6 @@ module ol_framework::proof_of_fee { // assert!(this_epoch > pof.last_epoch_retracted, error::ol_tx(EABOVE_RETRACT_LIMIT)); //////// LEAVE COMMENTED. Code for a potential upgrade. //////// - pof.epoch_expiration = 0; pof.bid = 0; pof.last_epoch_retracted = this_epoch; @@ -905,10 +908,8 @@ module ol_framework::proof_of_fee { assert!(reward == 105, 1003); assert!(clear_percent == 50, 1004); assert!(median_bid == 33, 1005); - } - // Scenario: The reward is too low during 5 days (short window). People are not bidding very high. #[test(vm = @ol_framework)] fun thermostat_increase_long(vm: signer) acquires ConsensusReward { @@ -930,7 +931,6 @@ module ol_framework::proof_of_fee { i = i + 1; }; - test_mock_reward( &vm, 100, @@ -953,10 +953,8 @@ module ol_framework::proof_of_fee { assert!(reward == 110, 1003); assert!(clear_percent == 50, 1004); assert!(median_bid == 33, 1005); - } - // Scenario: The reward is too high during 5 days (short window). People are bidding over 95% of the baseline fee. #[test(vm = @ol_framework)] fun thermostat_decrease_short(vm: signer) acquires ConsensusReward { @@ -977,7 +975,6 @@ module ol_framework::proof_of_fee { i = i + 1; }; - test_mock_reward( &vm, 100, @@ -1000,10 +997,9 @@ module ol_framework::proof_of_fee { assert!(reward == 95, 1000); assert!(clear_percent == 50, 1004); assert!(median_bid == 33, 1005); - } - // Scenario: The reward is too low during 5 days (short window). People are not bidding very high. + // Scenario: The reward is too low during 5 days (short window). People are not bidding very high. #[test(vm = @ol_framework)] fun thermostat_decrease_long(vm: signer) acquires ConsensusReward { use diem_framework::chain_id; @@ -1024,7 +1020,6 @@ module ol_framework::proof_of_fee { i = i + 1; }; - test_mock_reward( &vm, 100, @@ -1047,7 +1042,6 @@ module ol_framework::proof_of_fee { assert!(reward == 90, 1003); assert!(clear_percent == 50, 1004); assert!(median_bid == 33, 1005); - } // #[test(vm = @ol_framework)] @@ -1063,4 +1057,306 @@ module ol_framework::proof_of_fee { // validator_universe::is_in_universe(@0x123); // } + + + // Calculate Final Set Size tests + + #[test] + fun test_calculate_final_set_size_boot_up_happy_day() { + // Happy Day: test complete boot up with plenty qualified bidders over multiple epochs + // having validators always compliant + + // Epoch 1 + let outgoing_compliant = 0; + let qualified_bidders = 100; + let mc_set_size = 4; + let result = calculate_final_set_size(outgoing_compliant, qualified_bidders, mc_set_size); + assert!(result == 4, 7357023); + + // Epoch 2 + outgoing_compliant = 4; + mc_set_size = 5; + result = calculate_final_set_size(outgoing_compliant, qualified_bidders, mc_set_size); + // 4 + (4/2 - 1) = 5 + assert!(result == 5, 7357024); + + // Epoch 3 + outgoing_compliant = 5; + mc_set_size = 6; + result = calculate_final_set_size(outgoing_compliant, qualified_bidders, mc_set_size); + // 5 + (5/2 - 1) = 6 + assert!(result == 6, 7357025); + + // Epoch 4 + outgoing_compliant = 6; + mc_set_size = 7; + result = calculate_final_set_size(outgoing_compliant, qualified_bidders, mc_set_size); + // 6 + (6/2 - 1) = 8 + assert!(result == 8, 7357026); + + // Epoch 5 + outgoing_compliant = 8; + mc_set_size = 9; + result = calculate_final_set_size(outgoing_compliant, qualified_bidders, mc_set_size); + // 8 + (8/2 - 1) = 11 + assert!(result == 11, 7357027); + + // Epoch 6 + outgoing_compliant = 11; + mc_set_size = 12; + result = calculate_final_set_size(outgoing_compliant, qualified_bidders, mc_set_size); + // 11 + (11/2 - 1) = 15 + assert!(result == 15, 7357028); + + // Epoch 6 + outgoing_compliant = 15; + mc_set_size = 16; + result = calculate_final_set_size(outgoing_compliant, qualified_bidders, mc_set_size); + // 15 + (15/2 - 1) = 21 + // min (21, 21) + assert!(result == 21, 7357028); + + // Epoch 7 - Boot up ended + outgoing_compliant = 21; + mc_set_size = 22; + result = calculate_final_set_size(outgoing_compliant, qualified_bidders, mc_set_size); + // min(22, 100*90%) = 22 + assert!(result == 22, 7357029); + } + + #[test] + fun test_calculate_final_set_size_boot_up_threshold() { + let qualified_bidders = 100; + + // Test boot up increases maximum set size to 21 + let outgoing_compliant = 20; + let mc_set_size = 20; + let result = calculate_final_set_size(outgoing_compliant, qualified_bidders, mc_set_size); + assert!(result == 21, 7357030); + + // Test set size stays at 21 + outgoing_compliant = 21; + mc_set_size = 21; + result = calculate_final_set_size(outgoing_compliant, qualified_bidders, mc_set_size); + assert!(result == 21, 7357030); + + // Test set size increases to 22 + outgoing_compliant = 22; + mc_set_size = 22; + result = calculate_final_set_size(outgoing_compliant, qualified_bidders, mc_set_size); + // min (22, 100*90%) = 22 + assert!(result == 22, 7357030); + } + + #[test] + fun test_calculate_final_set_size_competitive_set_no_changes() { + let outgoing_compliant = 21; + let qualified_bidders = 40; + let mc_set_size = 21; + let result = calculate_final_set_size(outgoing_compliant, qualified_bidders, mc_set_size); + assert!(result == 21, 7357030); + } + + #[test] + fun test_calculate_final_set_size_competitive_set_increases() { + let outgoing_compliant = 30; + let qualified_bidders = 40; + let mc_set_size = 31; + let result = calculate_final_set_size(outgoing_compliant, qualified_bidders, mc_set_size); + assert!(result == 31, 7357030); + } + + #[test] + fun test_calculate_final_set_size_competitive_set_decreases() { + let outgoing_compliant = 50; + let qualified_bidders = 50; + let mc_set_size = 50; + let result = calculate_final_set_size(outgoing_compliant, qualified_bidders, mc_set_size); + assert!(result == 40, 7357030); + } + + #[test] + fun test_calculate_final_set_size_competitive_set_decreases_to_boot_up() { + let outgoing_compliant = 21; + let qualified_bidders = 21; + let mc_set_size = 21; + let result = calculate_final_set_size(outgoing_compliant, qualified_bidders, mc_set_size); + // min(21, 21*80%) = 17 + assert!(result == 17, 7357030); + + let outgoing_compliant = 21; + let qualified_bidders = 20; + let mc_set_size = 21; + let result = calculate_final_set_size(outgoing_compliant, qualified_bidders, mc_set_size); + // mc value + assert!(result == 21, 7357030); + } + + // Tests for calculate_reward_adjustment + + #[test] + public fun cra_nominal_reward_zero() { + let median_history = vector::empty(); + let nominal_reward = 0; + + let (did_run, did_increment, amount) = calculate_reward_adjustment(&median_history, nominal_reward); + assert!(did_run == false, 7357001); + assert!(did_increment == false, 7357002); + assert!(amount == 0, 7357003); + } + + #[test] + public fun cra_empty_bid_history() { + let median_history = vector::empty(); + let nominal_reward = 1000; + + let (did_run, did_increment, amount) = calculate_reward_adjustment(&median_history, nominal_reward); + assert!(did_run == true, 7357004); + assert!(did_increment == false, 7357005); + assert!(amount == 0, 7357006); + } + + #[test] + public fun cra_less_than_16_bids() { + // 10 entries all with value 600 + let median_history = vector[600, 600, 600, 600, 600, 600, 600, 600, 600, 600]; + let nominal_reward = 1000; + + let (did_run, did_increment, amount) = calculate_reward_adjustment(&median_history, nominal_reward); + assert!(did_run == true, 7357007); + assert!(did_increment == false, 7357008); + assert!(amount == 0, 7357009); + } + + #[test] + public fun cra_exactly_16_bids() { + // 16 entries all with value 600 + let median_history = vector[600, 600, 600, 600, 600, 600, 600, 600, 600, 600, 600, 600, 600, 600, 600, 600]; + let nominal_reward = 1000; + + let (did_run, did_increment, amount) = calculate_reward_adjustment(&median_history, nominal_reward); + assert!(did_run == true, 7357010); + assert!(did_increment == false, 7357011); + assert!(amount == 0, 7357012); + } + + #[test] + public fun cra_more_than_16_bids() { + // 20 entries all with value 600 + let median_history = vector[600, 600, 600, 600, 600, 600, 600, 600, 600, 600, 600, 600, 600, 600, 600, 600, 600, 600, 600, 600]; + let nominal_reward = 1000; + + let (did_run, did_increment, amount) = calculate_reward_adjustment(&median_history, nominal_reward); + assert!(did_run == true, 7357013); + assert!(did_increment == false, 7357014); + assert!(amount == 0, 7357015); + } + + #[test] + public fun cra_all_bids_above_upper_bound_short_window() { + // 6 entries all with value 960 + let median_history = vector[960, 960, 960, 960, 960, 960]; + let nominal_reward = 1000; + + let (did_run, did_increment, amount) = calculate_reward_adjustment(&median_history, nominal_reward); + assert!(did_run == true, 7357016); + assert!(did_increment == false, 7357017); + assert!(amount == nominal_reward / 20, 7357018); + } + + #[test] + public fun cra_all_bids_above_upper_bound_long_window() { + // 11 entries all with value 960 + let median_history = vector[960, 960, 960, 960, 960, 960, 960, 960, 960, 960, 960]; + let nominal_reward = 1000; + + let (did_run, did_increment, amount) = calculate_reward_adjustment(&median_history, nominal_reward); + assert!(did_run == true, 7357019); + assert!(did_increment == false, 7357020); + assert!(amount == nominal_reward / 10, 7357021); + } + + #[test] + public fun cra_all_bids_below_lower_bound_short_window() { + // 6 entries all with value 400 + let median_history = vector[400, 400, 400, 400, 400, 400]; + let nominal_reward = 1000; + + let (did_run, did_increment, amount) = calculate_reward_adjustment(&median_history, nominal_reward); + assert!(did_run == true, 7357022); + assert!(did_increment == true, 7357023); + assert!(amount == nominal_reward / 20, 7357024); + } + + #[test] + public fun cra_all_bids_below_lower_bound_long_window() { + // 11 entries all with value 400 + let median_history = vector[400, 400, 400, 400, 400, 400, 400, 400, 400, 400, 400]; + let nominal_reward = 1000; + + let (did_run, did_increment, amount) = calculate_reward_adjustment(&median_history, nominal_reward); + assert!(did_run == true, 7357025); + assert!(did_increment == true, 7357026); + assert!(amount == nominal_reward / 10, 7357027); + } + + #[test] + public fun cra_mixed_bids_with_majority_above() { + // 9 entries above upper bound and 7 entries below lower bound + let median_history = vector[960, 960, 960, 960, 960, 960, 960, 960, 960, 400, 400, 400, 400, 400, 400, 400]; + let nominal_reward = 1000; + + let (did_run, did_increment, amount) = calculate_reward_adjustment(&median_history, nominal_reward); + assert!(did_run == true, 7357028); + assert!(did_increment == false, 7357029); + assert!(amount == nominal_reward / 20, 7357030); // Since the total entries are 16, it falls under the short window + } + + #[test] + public fun cra_mixed_bids_with_majority_below() { + // 9 entries below lower bound and 7 entries above upper bound + let median_history = vector[400, 400, 400, 400, 400, 400, 400, 400, 400, 960, 960, 960, 960, 960, 960, 960]; + let nominal_reward = 1000; + + let (did_run, did_increment, amount) = calculate_reward_adjustment(&median_history, nominal_reward); + assert!(did_run == true, 7357031); + assert!(did_increment == true, 7357032); + assert!(amount == nominal_reward / 20, 7357033); // Since the total entries are 16, it falls under the short window + } + + #[test] + public fun cra_mixed_bids_without_clear_majority() { + // 4 entries below lower bound and 4 entries above upper bound + let median_history = vector[400, 400, 400, 400, 960, 960, 960, 960]; + let nominal_reward = 1000; + + let (did_run, did_increment, amount) = calculate_reward_adjustment(&median_history, nominal_reward); + assert!(did_run == true, 7357034); + assert!(did_increment == false, 7357035); + assert!(amount == 0, 7357036); + } + + #[test] + public fun cra_majority_above_long_window() { + // 12 entries above upper bound and 4 entries below lower bound + let median_history = vector[960, 960, 960, 960, 960, 960, 960, 960, 960, 960, 960, 960, 400, 400, 400, 400]; + let nominal_reward = 1000; + + let (did_run, did_increment, amount) = calculate_reward_adjustment(&median_history, nominal_reward); + assert!(did_run == true, 7357037); + assert!(did_increment == false, 7357038); + assert!(amount == nominal_reward / 10, 7357039); // Majority above and longer than long window + } + + #[test] + public fun cra_majority_above_short_window() { + // 7 entries above upper bound and 4 entries below lower bound + let median_history = vector[960, 960, 960, 960, 960, 960, 960, 400, 400, 400, 400]; + let nominal_reward = 1000; + + let (did_run, did_increment, amount) = calculate_reward_adjustment(&median_history, nominal_reward); + assert!(did_run == true, 7357040); + assert!(did_increment == false, 7357041); + assert!(amount == nominal_reward / 20, 7357042); // Majority above and longer than short window but not long window + } } diff --git a/framework/libra-framework/sources/ol_sources/rank.todo b/framework/libra-framework/sources/ol_sources/rank.todo index d6527dab2..d7e869004 100644 --- a/framework/libra-framework/sources/ol_sources/rank.todo +++ b/framework/libra-framework/sources/ol_sources/rank.todo @@ -82,7 +82,7 @@ module ol_framework::rank{ }; // Sorting the accounts vector based on value (weights). - // Bubble sort algorithm + // Bubble sort algorithm => use address_utils::sort_by_values let len_filtered = vector::length(&weights); if (len_filtered < 2) return weights; let i = 0; diff --git a/framework/libra-framework/sources/ol_sources/tests/boundary.test.move b/framework/libra-framework/sources/ol_sources/tests/boundary.test.move index d89036ca8..659f207d1 100644 --- a/framework/libra-framework/sources/ol_sources/tests/boundary.test.move +++ b/framework/libra-framework/sources/ol_sources/tests/boundary.test.move @@ -4,6 +4,13 @@ module ol_framework::test_boundary { use std::vector; use std::features; use diem_std::bls12381; + use diem_framework::stake; + use diem_framework::timestamp; + use diem_framework::reconfiguration; + use diem_framework::diem_governance; + use diem_framework::transaction_fee; + use diem_framework::account; + use ol_framework::burn; use ol_framework::mock; use ol_framework::proof_of_fee; use ol_framework::jail; @@ -14,12 +21,8 @@ module ol_framework::test_boundary { use ol_framework::epoch_boundary; use ol_framework::block; use ol_framework::ol_account; - use diem_framework::stake; - use diem_framework::reconfiguration; - use diem_framework::timestamp; - use diem_framework::diem_governance; - // use diem_std::debug::print; + use diem_std::debug::print; const Alice: address = @0x1000a; const Bob: address = @0x1000b; @@ -46,7 +49,13 @@ module ol_framework::test_boundary { // We need to test e2e of the epoch boundary #[test(root = @ol_framework)] fun e2e_boundary_happy(root: signer) { - let _vals = common_test_setup(&root); + let vals = common_test_setup(&root); + + // check vals balance + vector::for_each(vals, |addr| { + let (_unlocked, total) = ol_account::balance(addr); + assert!(total == 500_000, 7357000); + }); mock::trigger_epoch(&root); @@ -56,17 +65,28 @@ module ol_framework::test_boundary { // all validators were compliant, should be +1 of the 10 vals assert!(epoch_boundary::get_seats_offered() == 11, 7357002); - // all vals had winning bids, but it was less than the seats on offer assert!(vector::length(&epoch_boundary::get_auction_winners()) == 10, 7357003); // all of the auction winners became the validators ulitmately assert!(vector::length(&epoch_boundary::get_actual_vals()) == 10, 7357004); + + // check vals rewards received and bid fees collected + // balance + reward - fee = 500_000 + 1_000_000 - 10_000 = 1_490_000 + vector::for_each(vals, |addr| { + let (_unlocked, total) = ol_account::balance(addr); + assert!(total == 1_499_000, 7357005); + }); + + // check subsidy for new rewards and fees collected + // fees collected = 10 * 1_000_000 + 10 * 1_000 = 10_010_000 + print(&transaction_fee::system_fees_collected()); + assert!(transaction_fee::system_fees_collected() == 10_010_000, 7357006); } #[test(root = @ol_framework, alice = @0x1000a, marlon_rando = @0x12345)] fun e2e_add_validator_happy(root: signer, alice: signer, marlon_rando: signer) { - let _vals = common_test_setup(&root); + let initial_vals = common_test_setup(&root); // generate credentials for validator registration ol_account::transfer(&alice, @0x12345, 200000); @@ -77,33 +97,85 @@ module ol_framework::test_boundary { // MARLON PAYS THE BID let vals = validator_universe::get_eligible_validators(); - assert!(vector::length(&vals) == 11, 7357000); + assert!(vector::length(&vals) == 11, 7357001); + mock::mock_bids(&vals); // MARLON HAS MANY FRIENDS vouch::test_set_buddies(@0x12345, vals); let (errs, _pass) = proof_of_fee::audit_qualification(@0x12345); - assert!(vector::length(&errs) == 0, 7357001); + assert!(vector::length(&errs) == 0, 7357002); + + // get initial vals balance + let balances = vector::map(initial_vals, |addr| { + let (_unlocked, total) = ol_account::balance(addr); + total + }); mock::trigger_epoch(&root); - assert!(epoch_boundary::get_reconfig_success(), 7357002); + assert!(epoch_boundary::get_reconfig_success(), 7357003); // all validators were compliant, should be +1 of the 10 vals - assert!(epoch_boundary::get_seats_offered() == 11, 7357003); - + assert!(epoch_boundary::get_seats_offered() == 11, 7357004); // NOTE: now MARLON is INCLUDED in this, and we filled all the seats on offer. // all vals had winning bids, but it was less than the seats on offer - assert!(vector::length(&epoch_boundary::get_auction_winners()) == 11, 7357003); + assert!(vector::length(&epoch_boundary::get_auction_winners()) == 11, 7357005); // all of the auction winners became the validators ulitmately - assert!(vector::length(&epoch_boundary::get_actual_vals()) == 11, 7357004); + assert!(vector::length(&epoch_boundary::get_actual_vals()) == 11, 7357006); + + // check initial vals rewards received and bid fees collected + // previous balance = current - reward + fee + let i = 0; + while (i < vector::length(&initial_vals)) { + let (_unlocked, current) = ol_account::balance(*vector::borrow(&initial_vals, i)); + let previous = *vector::borrow(&balances, i); + assert!(current == (previous + 1_000_000 - 1_000), 7357007); + i = i + 1; + }; + + // check Marlon's balance: 200_000 - 1_000 = 199_000 + let (_unlocked, marlon_balance) = ol_account::balance(@0x12345); + assert!(marlon_balance == 199_000, 7357008); + + // check subsidy for new rewards and fees collected + // fees collected = 11 * 1_000_000 + 11 * 1_000 = 11_011_000 + assert!(transaction_fee::system_fees_collected() == 11_011_000, 7357009); // another epoch and everyone is compliant as well mock::mock_all_vals_good_performance(&root); mock::trigger_epoch(&root); - assert!(epoch_boundary::get_seats_offered() == 12, 7357005); - assert!(vector::length(&epoch_boundary::get_actual_vals()) == 11, 7357006); + + assert!(epoch_boundary::get_seats_offered() == 12, 7357010); + assert!(vector::length(&epoch_boundary::get_actual_vals()) == 11, 7357011); + + // check initial vals rewards received and bid fees collected + // previous balance = current - 2*reward + 2*fee + let i = 0; + while (i < vector::length(&initial_vals)) { + let (_unlocked, current) = ol_account::balance(*vector::borrow(&initial_vals, i)); + let previous = *vector::borrow(&balances, i); + assert!(current == (previous + 2_000_000 - 2_000), 7357012); + i = i + 1; + }; + + // check Marlon's balance: 200_000 + 1_000_000 - 2_000 = 1_198_000 + let (_unlocked, marlon_balance) = ol_account::balance(@0x12345); + assert!(marlon_balance == 1_198_000, 7357013); + + // CHECK BURNT FEES + + // prepare clean epoch + mock::trigger_epoch(&root); + + let (before, _) = burn::get_lifetime_tracker(); + + mock::trigger_epoch(&root); + + // check that only the entry fee sum is being burnt + let (after,_) = burn::get_lifetime_tracker(); + assert!(after - before == 11_000_000, 7357014); // scale 1_000 } #[test(root = @ol_framework, alice = @0x1000a, marlon_rando = @0x12345)] @@ -140,16 +212,42 @@ module ol_framework::test_boundary { fun e2e_boundary_excludes_jail(root: signer) { let vals = common_test_setup(&root); let alice_addr = *vector::borrow(&vals, 0); - jail::jail(&root, alice_addr); - assert!(jail::is_jailed(alice_addr), 7357000); + + // mock vals performance + let i = 1; + while (i < vector::length(&vals)) { + let addr = *vector::borrow(&vals, i); + stake::mock_performance(&root, addr, 10, 0); + i = i + 1; + }; + + // make Alice val not compliant to end up in jail + stake::mock_performance(&root, alice_addr, 10, 10); + + // get Alice balance before epoch boundary + let (_unlocked, alice_before) = ol_account::balance(alice_addr); + + // new epoch mock::trigger_epoch(&root); + // check that Alice is jailed + assert!(jail::is_jailed(alice_addr), 7357001); + + // ensure Alice did not receive rewards + let (_unlocked, alice_after) = ol_account::balance(alice_addr); + assert!(alice_before == alice_after, 7357002); + + // check that validator set reduced by 1 let qualified_bidders = epoch_boundary::get_qualified_bidders(); - assert!(vector::length(&qualified_bidders) == (vector::length(&vals) - 1), 7357003); + assert!(vector::length(&qualified_bidders) == 9, 7357003); + + // check subsidy for new rewards and fees collected + // fees collected = 9 * 1_000_000 + 9 * 2_000 = 9_018_000 + assert!(transaction_fee::system_fees_collected() == 9_018_000, 7357004); // all vals had winning bids, but it was less than the seats on offer - assert!(vector::length(&epoch_boundary::get_auction_winners()) == vector::length(&qualified_bidders) , 7357003); - assert!(epoch_boundary::get_reconfig_success(), 7357001); + assert!(vector::length(&epoch_boundary::get_auction_winners()) == vector::length(&qualified_bidders) , 7357005); + assert!(epoch_boundary::get_reconfig_success(), 7357006); } #[test(root = @ol_framework, marlon = @0x12345)] @@ -178,6 +276,65 @@ module ol_framework::test_boundary { assert!(epoch == 4, 7357003); } + #[test(root = @ol_framework)] + fun epoch_increase_thermostat(root: &signer) { + let vals = common_test_setup(root); + + // mock bids + vector::for_each(vals, |a| { + let sig = account::create_signer_for_test(a); + proof_of_fee::pof_update_bid(&sig, 0100, 30); // 10% for 30 epochs + }); + + // mock bid history to increase thermostat by 5% + proof_of_fee::test_mock_reward( + root, + 1000, // nominal reward + 0100, // clearing bid + 0100, // median win bid + vector[ 0100, 0100, 0100, 0100, 0100, 0100 ] // median history bellow 50% + ); + + // trigger epoch + mock::trigger_epoch(root); + + // check subsidy increased by 5% + // fees collected = entry fee + reward * 105% + // entry fee = 100 * 10 = 1_000 + // reward = 1050 * 10 = 10_500 + // fees collected = 11_500 + assert!(transaction_fee::system_fees_collected() == 11_500, 7357001); + } + + #[test(root = @ol_framework)] + fun epoch_decrease_thermostat(root: &signer) { + let vals = common_test_setup(root); + + // mock bids + vector::for_each(vals, |a| { + let sig = account::create_signer_for_test(a); + proof_of_fee::pof_update_bid(&sig, 0970, 30); // 97% for 30 epochs + }); + + // mock bid history to decrease thermostat by 5% + proof_of_fee::test_mock_reward( + root, + 100_000, // nominal reward + 0970, // clearing bid + 0970, // median win bid + vector[ 0970, 0970, 0970, 0970, 0970, 0970 ] // median history above 95% + ); + + // trigger epoch + mock::trigger_epoch(root); + + // check subsidy decreased by 5% + // fees collected = entry fee + reward * 95% + // entry fee = 97_000 * 10 = 970_000 + // reward = 95_000 * 10 = 950_000 + // fees collected = 1_920_000 + assert!(transaction_fee::system_fees_collected() == 1_920_000, 7357001); + } #[test(root = @ol_framework, marlon = @0x12345)] #[expected_failure(abort_code = 2, location = 0x1::epoch_boundary)] diff --git a/framework/libra-framework/sources/ol_sources/tests/burn.test.move b/framework/libra-framework/sources/ol_sources/tests/burn.test.move index 2f79941ad..51c66a9f6 100644 --- a/framework/libra-framework/sources/ol_sources/tests/burn.test.move +++ b/framework/libra-framework/sources/ol_sources/tests/burn.test.move @@ -66,303 +66,297 @@ module ol_framework::test_burn { } - #[test(root = @ol_framework, alice = @0x1000a, bob = @0x1000d, eve = @0x1000e)] - fun burn_to_match_index(root: &signer, alice: &signer, bob: &signer, eve: &signer) { - // Scenario: - // There are two community wallets. Alice and Bob's - // EVE donates to both. But mostly to Alice. - // The Match Index, should reflect that. - - let vals = mock::genesis_n_vals(root, 5); // need to include eve to init funds - mock::ol_initialize_coin_and_fund_vals(root, 1000000, true); - // start at epoch 1, since turnout tally needs epoch info, and 0 may cause issues - mock::trigger_epoch(root); - - let (communityA, _cap) = ol_account::test_ol_create_resource_account(alice, b"0x1"); - let addr_A = signer::address_of(&communityA); - community_wallet_init::init_community(&communityA, vals, 2); // make all the vals signers - - let (communityB, _cap) = ol_account::test_ol_create_resource_account(bob, b"0xdeadbeef"); - let addr_B = signer::address_of(&communityB); - community_wallet_init::init_community(&communityB, vals, 2); // make all the vals signers - - let eve_donation_to_A = 75; - ol_account::transfer(eve, addr_A, eve_donation_to_A); - let (_, _, total_funds_sent_by_eve) = receipts::read_receipt(signer::address_of(eve), addr_A); - assert!(total_funds_sent_by_eve == eve_donation_to_A, 7357001); - - let eve_donation_to_B = 25; - ol_account::transfer(eve, addr_B, eve_donation_to_B); - let (_, _, total_funds_sent_by_eve) = receipts::read_receipt(signer::address_of(eve), addr_B); - assert!(total_funds_sent_by_eve == eve_donation_to_B, 7357002); - - let qualifying = vector::singleton(addr_A); - vector::push_back(&mut qualifying, addr_B); - - // simulate index recalculation at epoch boundary - match_index::test_reset_ratio(root, qualifying); - let (_addrs_vec, _index_vec, ratio_vec) = match_index::get_ratios(); - - // so far addr_A should have 75 coins from Eve's transfer - let(_, balance_pre) = ol_account::balance(addr_A); - assert!(balance_pre == eve_donation_to_A, 7357003); - - // let's burn some of bob's coins - // first he sets his burn preference - burn::set_send_community(bob, true); - let bob_burn = 100000; - let c = ol_account::withdraw(bob, bob_burn); - // then the system burns based on his preference. - burn::test_vm_burn_with_user_preference(root, signer::address_of(bob), c); - // check that a recycle to match index happened instead of a straight burn - let (_lifetime_burn, lifetime_match) = burn::get_lifetime_tracker(); - assert!(lifetime_match > 0, 7357003); - - - // the balance of address A should be increase by the ratio above 75% * bob's burn of 100_000 - let ratio_A = vector::borrow(&ratio_vec, 0); - let bob_burn_share_A = fixed_point32::multiply_u64(bob_burn, *ratio_A); - - let(_, balance) = ol_account::balance(addr_A); - assert!(balance > balance_pre, 7357004); - assert!(balance == (bob_burn_share_A + eve_donation_to_A), 7357005); - } - - - #[test(root = @ol_framework, alice = @0x1000a, bob = @0x1000d, eve = @0x1000e)] - fun simple_burn(root: &signer, alice: &signer, bob: &signer, eve: &signer) { - // Scenario: - // There are two community wallets. Alice and Bob's - // EVE donates to both. But mostly to Alice. - // The Match Index, should reflect that. - - let vals = mock::genesis_n_vals(root, 5); // need to include eve to init funds - mock::ol_initialize_coin_and_fund_vals(root, 1000000, true); - // start at epoch 1, since turnout tally needs epoch info, and 0 may cause issues - mock::trigger_epoch(root); - - let (communityA, _cap) = ol_account::test_ol_create_resource_account(alice, b"0x1"); - let addr_A = signer::address_of(&communityA); - community_wallet_init::init_community(&communityA, vals, 2); // make all the vals signers - - let (communityB, _cap) = ol_account::test_ol_create_resource_account(bob, b"0xdeadbeef"); - let addr_B = signer::address_of(&communityB); - community_wallet_init::init_community(&communityB, vals, 2); // make all the vals signers - - let eve_donation_to_A = 75; - ol_account::transfer(eve, addr_A, eve_donation_to_A); - let (_, _, total_funds_sent_by_eve) = receipts::read_receipt(signer::address_of(eve), addr_A); - assert!(total_funds_sent_by_eve == eve_donation_to_A, 7357001); - - let eve_donation_to_B = 25; - ol_account::transfer(eve, addr_B, eve_donation_to_B); - let (_, _, total_funds_sent_by_eve) = receipts::read_receipt(signer::address_of(eve), addr_B); - assert!(total_funds_sent_by_eve == eve_donation_to_B, 7357002); - - let qualifying = vector::singleton(addr_A); - vector::push_back(&mut qualifying, addr_B); - - // simulate index recalculation at epoch boundary - match_index::test_reset_ratio(root, qualifying); - // let (_addrs_vec, _index_vec, ratio_vec) = match_index::get_ratios(); - - // so far addr_A should have 75 coins from Eve's transfer - let(_, balance_pre) = ol_account::balance(addr_A); - assert!(balance_pre == eve_donation_to_A, 7357003); - - // let's burn some of bob's coins - // first he sets his burn preference - // IMPORTANT: BOB IS SETTING PREFERENCE TO SIMPLE BURN - burn::set_send_community(bob, false); - let bob_burn = 100000; - let c = ol_account::withdraw(bob, bob_burn); - // then the system burns based on his preference. - burn::test_vm_burn_with_user_preference(root, signer::address_of(bob), c); - - // no match recycling happened - let (lifetime_burn, lifetime_match) = burn::get_lifetime_tracker(); - assert!(lifetime_match == 0, 7357003); - assert!(lifetime_burn > 0, 7357004); - - - // NONE OF THE MATCH INDEX ACCOUNTS HAVE ANY CHANGE - let(_, balance) = ol_account::balance(addr_A); - assert!(balance == balance_pre, 7357005); - assert!(balance == eve_donation_to_A, 7357006); - } - - #[test(root = @ol_framework)] - fun epoch_fees_burn(root: &signer) { - // Scenario: - // the network starts. - // we mock some tx fees going into the collection - // the epoch turns and the fee account should be empty - // and the supply should be lower - - let n_vals = 5; - let _vals = mock::genesis_n_vals(root, n_vals); // need to include eve to init funds - let genesis_mint = 1000000; // 1 coin per - let epoch_reward = genesis_mint; // just to be explicit - - mock::ol_initialize_coin_and_fund_vals(root, genesis_mint, true); - let supply_pre = libra_coin::supply(); - let mocked_tx_fees = 1000000 * 100; // 100 coins in tx fee account - // 105 coins total - assert!(supply_pre == mocked_tx_fees + (n_vals * genesis_mint), 73570001); - - let fees = transaction_fee::system_fees_collected(); - assert!(fees == mocked_tx_fees, 73570002); - // start at epoch 1. NOTE Validators WILL GET PAID FOR EPOCH 1 - mock::trigger_epoch(root); // under 1000 rounds we don't - // evaluate performance of validators - let fees = transaction_fee::system_fees_collected(); - // There should be no fees left in tx fee wallet - assert!(fees == 0, 73570003); - - let validator_rewards = epoch_reward * n_vals; - - // of the initial supply, - // we expect to burn everything which was in the TX FEEs AFTER PAYING VALIDATOR REWARDS - let amount_burned_excess_tx_account = mocked_tx_fees - validator_rewards; - - // So the current supply should be lower, - // The the old supply was reduced by what was burned (the excess in tx bucket) - - let supply_post = libra_coin::supply(); - - assert!(supply_post == supply_pre - amount_burned_excess_tx_account, 73570003); - - // this ALSO means that the only supply left in this example - // is the rewards to validators from genesis, and from epoch 1 - assert!(supply_post == validator_rewards + (n_vals * genesis_mint), 73570003); - - } - - - #[test(root = @ol_framework, alice = @0x1000a, bob = @0x1000d, eve = @0x1000e)] - fun epoch_fees_match(root: &signer, alice: &signer, bob: &signer, eve: &signer) { - // Scenario: - // There are two community wallets. Alice and Bob's - // EVE donates to both. But mostly to Alice. - // The Match Index, should reflect that. - - let vals = mock::genesis_n_vals(root, 5); // need to include eve to init funds - mock::ol_initialize_coin_and_fund_vals(root, 1000000, true); - // start at epoch 1, since turnout tally needs epoch info, and 0 may cause issues - mock::trigger_epoch(root); - - let (communityA, _cap) = ol_account::test_ol_create_resource_account(alice, b"0x1"); - let addr_A = signer::address_of(&communityA); - community_wallet_init::init_community(&communityA, vals, 2); // make all the vals signers - - let (communityB, _cap) = ol_account::test_ol_create_resource_account(bob, b"0xdeadbeef"); - let addr_B = signer::address_of(&communityB); - community_wallet_init::init_community(&communityB, vals, 2); // make all the vals signers - - let eve_donation_to_A = 75; - ol_account::transfer(eve, addr_A, eve_donation_to_A); - let (_, _, total_funds_sent_by_eve) = receipts::read_receipt(signer::address_of(eve), addr_A); - assert!(total_funds_sent_by_eve == eve_donation_to_A, 7357001); - - let eve_donation_to_B = 25; - ol_account::transfer(eve, addr_B, eve_donation_to_B); - let (_, _, total_funds_sent_by_eve) = receipts::read_receipt(signer::address_of(eve), addr_B); - assert!(total_funds_sent_by_eve == eve_donation_to_B, 7357002); - - let qualifying = vector::singleton(addr_A); - vector::push_back(&mut qualifying, addr_B); - - // simulate index recalculation at epoch boundary - match_index::test_reset_ratio(root, qualifying); - let (_addrs_vec, _index_vec, ratio_vec) = match_index::get_ratios(); - - // so far addr_A should have 75 coins from Eve's transfer - let(_, balance_pre) = ol_account::balance(addr_A); - assert!(balance_pre == eve_donation_to_A, 7357003); - - // let's burn some of bob's coins - // first he sets his burn preference - burn::set_send_community(bob, true); - let bob_burn = 100000; - let c = ol_account::withdraw(bob, bob_burn); - // then the system burns based on his preference. - burn::test_vm_burn_with_user_preference(root, signer::address_of(bob), c); - // check that a recycle to match index happened instead of a straight burn - let (_lifetime_burn, lifetime_match) = burn::get_lifetime_tracker(); - assert!(lifetime_match > 0, 7357003); + #[test(root = @ol_framework, alice = @0x1000a, bob = @0x1000d, eve = @0x1000e)] + fun burn_to_match_index(root: &signer, alice: &signer, bob: &signer, eve: &signer) { + // Scenario: + // There are two community wallets. Alice and Bob's + // EVE donates to both. But mostly to Alice. + // The Match Index, should reflect that. + + let vals = mock::genesis_n_vals(root, 5); // need to include eve to init funds + mock::ol_initialize_coin_and_fund_vals(root, 1000000, true); + // start at epoch 1, since turnout tally needs epoch info, and 0 may cause issues + mock::trigger_epoch(root); + + let (communityA, _cap) = ol_account::test_ol_create_resource_account(alice, b"0x1"); + let addr_A = signer::address_of(&communityA); + community_wallet_init::init_community(&communityA, vals, 2); // make all the vals signers + + let (communityB, _cap) = ol_account::test_ol_create_resource_account(bob, b"0xdeadbeef"); + let addr_B = signer::address_of(&communityB); + community_wallet_init::init_community(&communityB, vals, 2); // make all the vals signers + + let eve_donation_to_A = 75; + ol_account::transfer(eve, addr_A, eve_donation_to_A); + let (_, _, total_funds_sent_by_eve) = receipts::read_receipt(signer::address_of(eve), addr_A); + assert!(total_funds_sent_by_eve == eve_donation_to_A, 7357001); + + let eve_donation_to_B = 25; + ol_account::transfer(eve, addr_B, eve_donation_to_B); + let (_, _, total_funds_sent_by_eve) = receipts::read_receipt(signer::address_of(eve), addr_B); + assert!(total_funds_sent_by_eve == eve_donation_to_B, 7357002); + + let qualifying = vector::singleton(addr_A); + vector::push_back(&mut qualifying, addr_B); + + // simulate index recalculation at epoch boundary + match_index::test_reset_ratio(root, qualifying); + let (_addrs_vec, _index_vec, ratio_vec) = match_index::get_ratios(); + + // so far addr_A should have 75 coins from Eve's transfer + let(_, balance_pre) = ol_account::balance(addr_A); + assert!(balance_pre == eve_donation_to_A, 7357003); + + // let's burn some of bob's coins + // first he sets his burn preference + burn::set_send_community(bob, true); + let bob_burn = 100000; + let c = ol_account::withdraw(bob, bob_burn); + // then the system burns based on his preference. + burn::test_vm_burn_with_user_preference(root, signer::address_of(bob), c); + // check that a recycle to match index happened instead of a straight burn + let (_lifetime_burn, lifetime_match) = burn::get_lifetime_tracker(); + assert!(lifetime_match > 0, 7357003); + + + // the balance of address A should be increase by the ratio above 75% * bob's burn of 100_000 + let ratio_A = vector::borrow(&ratio_vec, 0); + let bob_burn_share_A = fixed_point32::multiply_u64(bob_burn, *ratio_A); + + let(_, balance) = ol_account::balance(addr_A); + assert!(balance > balance_pre, 7357004); + assert!(balance == (bob_burn_share_A + eve_donation_to_A), 7357005); + } + + + #[test(root = @ol_framework, alice = @0x1000a, bob = @0x1000d, eve = @0x1000e)] + fun simple_burn(root: &signer, alice: &signer, bob: &signer, eve: &signer) { + // Scenario: + // There are two community wallets. Alice and Bob's + // EVE donates to both. But mostly to Alice. + // The Match Index, should reflect that. + + let vals = mock::genesis_n_vals(root, 5); // need to include eve to init funds + mock::ol_initialize_coin_and_fund_vals(root, 1000000, true); + // start at epoch 1, since turnout tally needs epoch info, and 0 may cause issues + mock::trigger_epoch(root); + + let (communityA, _cap) = ol_account::test_ol_create_resource_account(alice, b"0x1"); + let addr_A = signer::address_of(&communityA); + community_wallet_init::init_community(&communityA, vals, 2); // make all the vals signers + + let (communityB, _cap) = ol_account::test_ol_create_resource_account(bob, b"0xdeadbeef"); + let addr_B = signer::address_of(&communityB); + community_wallet_init::init_community(&communityB, vals, 2); // make all the vals signers + + let eve_donation_to_A = 75; + ol_account::transfer(eve, addr_A, eve_donation_to_A); + let (_, _, total_funds_sent_by_eve) = receipts::read_receipt(signer::address_of(eve), addr_A); + assert!(total_funds_sent_by_eve == eve_donation_to_A, 7357001); + + let eve_donation_to_B = 25; + ol_account::transfer(eve, addr_B, eve_donation_to_B); + let (_, _, total_funds_sent_by_eve) = receipts::read_receipt(signer::address_of(eve), addr_B); + assert!(total_funds_sent_by_eve == eve_donation_to_B, 7357002); + + let qualifying = vector::singleton(addr_A); + vector::push_back(&mut qualifying, addr_B); + + // simulate index recalculation at epoch boundary + match_index::test_reset_ratio(root, qualifying); + // let (_addrs_vec, _index_vec, ratio_vec) = match_index::get_ratios(); + + // so far addr_A should have 75 coins from Eve's transfer + let(_, balance_pre) = ol_account::balance(addr_A); + assert!(balance_pre == eve_donation_to_A, 7357003); + + // let's burn some of bob's coins + // first he sets his burn preference + // IMPORTANT: BOB IS SETTING PREFERENCE TO SIMPLE BURN + burn::set_send_community(bob, false); + let bob_burn = 100000; + let c = ol_account::withdraw(bob, bob_burn); + // then the system burns based on his preference. + burn::test_vm_burn_with_user_preference(root, signer::address_of(bob), c); + + // no match recycling happened + let (lifetime_burn, lifetime_match) = burn::get_lifetime_tracker(); + assert!(lifetime_match == 0, 7357003); + assert!(lifetime_burn > 0, 7357004); + + + // NONE OF THE MATCH INDEX ACCOUNTS HAVE ANY CHANGE + let(_, balance) = ol_account::balance(addr_A); + assert!(balance == balance_pre, 7357005); + assert!(balance == eve_donation_to_A, 7357006); + } + + #[test(root = @ol_framework)] + fun epoch_fees_burn(root: &signer) { + // Scenario: + // the network starts. + // we mock some tx fees going into the collection + // the epoch turns and the fee account should be empty + // and the supply should be lower + let n_vals = 5; + let _vals = mock::genesis_n_vals(root, n_vals); // need to include eve to init funds + let genesis_mint = 1_000_000; // 1 coin per + let epoch_reward = genesis_mint; // just to be explicit - // the balance of address A should be increase by the ratio above 75% * bob's burn of 100_000 - let ratio_A = vector::borrow(&ratio_vec, 0); - let bob_burn_share_A = fixed_point32::multiply_u64(bob_burn, *ratio_A); - - let(_, balance) = ol_account::balance(addr_A); - assert!(balance > balance_pre, 7357004); - assert!(balance == (bob_burn_share_A + eve_donation_to_A), 7357005); - } + mock::ol_initialize_coin_and_fund_vals(root, genesis_mint, true); + let supply_pre = libra_coin::supply(); + let bruce_fortune = 100_000_000_000; + let mocked_tx_fees = 5_000_000 * 100; // 100 coins in tx fee account + assert!(supply_pre == mocked_tx_fees + bruce_fortune + (n_vals * genesis_mint), 73570001); + let fees = transaction_fee::system_fees_collected(); + assert!(fees == mocked_tx_fees, 73570002); - // TODO: + // start at epoch 1. NOTE Validators WILL GET PAID FOR EPOCH 1 + mock::trigger_epoch(root); // under 1000 rounds we don't - // #[test(root = @ol_framework, alice = @0x1000a, bob = @0x1000d, eve = @0x1000e)] - // fun fee_makers_calc(root: &signer, alice: &signer, bob: &signer, eve: &signer) { - // // Scenario: - // assert!(TransactionFee::get_fees_collected()==0, 735701); - // let coin = Diem::mint(&vm, 1); - // TransactionFee::pay_fee_and_track(@Alice, coin); + // evaluate performance of validators + let fees = transaction_fee::system_fees_collected(); + // There should be only entry fee left in tx fee wallet + assert!(fees == 5_000_000, 73570003); - // let fee_makers = TransactionFee::get_fee_makers(); - // assert!(Vector::length(&fee_makers)==1, 735702); - // assert!(TransactionFee::get_fees_collected()==1, 735703); - // } + let validator_rewards = epoch_reward * n_vals; - #[test(root=@ol_framework, alice=@0x1000a)] - fun track_fees(root: &signer, alice: address) { - // use ol_framework::libra_coin; - let _vals = mock::genesis_n_vals(root, 1); // need to include eve to init funds - mock::ol_initialize_coin_and_fund_vals(root, 10000, true); + // of the initial supply, + // we expect to burn everything which was in the TX FEEs AFTER PAYING VALIDATOR REWARDS + let amount_burned_excess_tx_account = mocked_tx_fees - validator_rewards; - let marlon_rando = @0x12345; - ol_account::create_account(root, marlon_rando); + // So the current supply should be lower, + // The the old supply was reduced by what was burned (the excess in tx bucket) + let supply_post = libra_coin::supply(); + assert!(supply_post == supply_pre - amount_burned_excess_tx_account, 73570003); - let rando_money = 5; - let coin_option = ol_account::test_vm_withdraw(root, alice, rando_money); + // this ALSO means that the only supply left in this example + // is the rewards to validators from genesis, and from epoch 1 + // and the bruce fortune used to pledge infra escrow + assert!(supply_post == validator_rewards + (n_vals * genesis_mint) + bruce_fortune, 73570003); + } - if (option::is_some(&coin_option)) { - let c = option::extract(&mut coin_option); - // shortcut: we have the VM use alices money to pay a fee for marlon. - transaction_fee::vm_pay_fee(root, marlon_rando, c); - }; - option::destroy_none(coin_option); + #[test(root = @ol_framework, alice = @0x1000a, bob = @0x1000d, eve = @0x1000e)] + fun epoch_fees_match(root: &signer, alice: &signer, bob: &signer, eve: &signer) { + // Scenario: + // There are two community wallets. Alice and Bob's + // EVE donates to both. But mostly to Alice. + // The Match Index, should reflect that. + + let vals = mock::genesis_n_vals(root, 5); // need to include eve to init funds + mock::ol_initialize_coin_and_fund_vals(root, 1000000, true); + // start at epoch 1, since turnout tally needs epoch info, and 0 may cause issues + mock::trigger_epoch(root); + + let (communityA, _cap) = ol_account::test_ol_create_resource_account(alice, b"0x1"); + let addr_A = signer::address_of(&communityA); + community_wallet_init::init_community(&communityA, vals, 2); // make all the vals signers + + let (communityB, _cap) = ol_account::test_ol_create_resource_account(bob, b"0xdeadbeef"); + let addr_B = signer::address_of(&communityB); + community_wallet_init::init_community(&communityB, vals, 2); // make all the vals signers + + let eve_donation_to_A = 75; + ol_account::transfer(eve, addr_A, eve_donation_to_A); + let (_, _, total_funds_sent_by_eve) = receipts::read_receipt(signer::address_of(eve), addr_A); + assert!(total_funds_sent_by_eve == eve_donation_to_A, 7357001); + + let eve_donation_to_B = 25; + ol_account::transfer(eve, addr_B, eve_donation_to_B); + let (_, _, total_funds_sent_by_eve) = receipts::read_receipt(signer::address_of(eve), addr_B); + assert!(total_funds_sent_by_eve == eve_donation_to_B, 7357002); + + let qualifying = vector::singleton(addr_A); + vector::push_back(&mut qualifying, addr_B); + + // simulate index recalculation at epoch boundary + match_index::test_reset_ratio(root, qualifying); + let (_addrs_vec, _index_vec, ratio_vec) = match_index::get_ratios(); + + // so far addr_A should have 75 coins from Eve's transfer + let(_, balance_pre) = ol_account::balance(addr_A); + assert!(balance_pre == eve_donation_to_A, 7357003); + + // let's burn some of bob's coins + // first he sets his burn preference + burn::set_send_community(bob, true); + let bob_burn = 100000; + let c = ol_account::withdraw(bob, bob_burn); + // then the system burns based on his preference. + burn::test_vm_burn_with_user_preference(root, signer::address_of(bob), c); + // check that a recycle to match index happened instead of a straight burn + let (_lifetime_burn, lifetime_match) = burn::get_lifetime_tracker(); + assert!(lifetime_match > 0, 7357003); + + + // the balance of address A should be increase by the ratio above 75% * bob's burn of 100_000 + let ratio_A = vector::borrow(&ratio_vec, 0); + let bob_burn_share_A = fixed_point32::multiply_u64(bob_burn, *ratio_A); + + let(_, balance) = ol_account::balance(addr_A); + assert!(balance > balance_pre, 7357004); + assert!(balance == (bob_burn_share_A + eve_donation_to_A), 7357005); + } - let fees = transaction_fee::system_fees_collected(); - assert!(fees == 100000005, 735701); - // Fees will include the initialization by MOCK. ol_initialize_coin_and_fund_vals - - // marlon is the only fee maker (since genesis) - let fee_makers = fee_maker::get_fee_makers(); - // includes 0x1 which makes a deposit on - assert!(vector::length(&fee_makers)==1, 735702); - let marlon_fees_made = fee_maker::get_user_fees_made(marlon_rando); - assert!(marlon_fees_made == 5, 735703); + // TODO: - mock::trigger_epoch(root); - // should reset the fee counter on everything. - // TODO: no proof of fee was charged on alice. But this is changed in - // feature branch `coin-methods-hygiene` + // #[test(root = @ol_framework, alice = @0x1000a, bob = @0x1000d, eve = @0x1000e)] + // fun fee_makers_calc(root: &signer, alice: &signer, bob: &signer, eve: &signer) { + // // Scenario: + // assert!(TransactionFee::get_fees_collected()==0, 735701); + // let coin = Diem::mint(&vm, 1); + // TransactionFee::pay_fee_and_track(@Alice, coin); - let marlon_fees_made = fee_maker::get_user_fees_made(marlon_rando); - assert!(marlon_fees_made == 0, 735704); - let fee_makers = fee_maker::get_fee_makers(); - assert!(vector::length(&fee_makers)==0, 735705); - let fees = transaction_fee::system_fees_collected(); - assert!(fees == 0, 7357008); + // let fee_makers = TransactionFee::get_fee_makers(); + // assert!(Vector::length(&fee_makers)==1, 735702); + // assert!(TransactionFee::get_fees_collected()==1, 735703); + // } - } + #[test(root=@ol_framework, alice=@0x1000a)] + fun track_fees(root: &signer, alice: address) { + // use ol_framework::libra_coin; + let _vals = mock::genesis_n_vals(root, 1); // need to include eve to init funds + mock::ol_initialize_coin_and_fund_vals(root, 10000, true); + let marlon_rando = @0x12345; + ol_account::create_account(root, marlon_rando); + + let rando_money = 5; + let coin_option = ol_account::test_vm_withdraw(root, alice, rando_money); + + if (option::is_some(&coin_option)) { + let c = option::extract(&mut coin_option); + // shortcut: we have the VM use alices money to pay a fee for marlon. + transaction_fee::vm_pay_fee(root, marlon_rando, c); + }; + option::destroy_none(coin_option); + + let fees = transaction_fee::system_fees_collected(); + assert!(fees == 500_000_005, 7357001); + // Fees will include the initialization by MOCK. ol_initialize_coin_and_fund_vals + + // marlon is the only fee maker (since genesis) + let fee_makers = fee_maker::get_fee_makers(); + // includes 0x1 which makes a deposit on + assert!(vector::length(&fee_makers)==1, 7357002); + + let marlon_fees_made = fee_maker::get_user_fees_made(marlon_rando); + assert!(marlon_fees_made == 5, 7357003); + + mock::trigger_epoch(root); + + let marlon_fees_made = fee_maker::get_user_fees_made(marlon_rando); + assert!(marlon_fees_made == 0, 7357004); + let fee_makers = fee_maker::get_fee_makers(); + assert!(vector::length(&fee_makers) == 0, 7357005); + let fees = transaction_fee::system_fees_collected(); + assert!(fees == 1_000_000, 7357006); // val set entry fee + } #[test(root = @ol_framework, alice_val = @0x1000a)] fun test_init_burn_tracker(root: &signer, alice_val: &signer) { @@ -383,7 +377,6 @@ module ol_framework::test_burn { assert!(prev_balance == marlon_salary, 7357002); assert!(burn_at_last_calc == 0, 7357003); assert!(cumu_burn == 0, 7357004); - } #[test(root = @ol_framework, alice_val=@0x1000a)] diff --git a/framework/libra-framework/sources/ol_sources/tests/donor_voice.test.move b/framework/libra-framework/sources/ol_sources/tests/donor_voice.test.move index bff2ba214..2b39cd964 100644 --- a/framework/libra-framework/sources/ol_sources/tests/donor_voice.test.move +++ b/framework/libra-framework/sources/ol_sources/tests/donor_voice.test.move @@ -40,7 +40,7 @@ module ol_framework::test_donor_voice { // vals claim the offer multi_action::claim_offer(alice, donor_voice_address); multi_action::claim_offer(bob, donor_voice_address); - + //need to be caged to finalize donor directed workflow and release control of the account multi_action::finalize_and_cage(&resource_sig, 2); @@ -61,7 +61,7 @@ module ol_framework::test_donor_voice { // vals claim the offer multi_action::claim_offer(alice, donor_voice_address); - multi_action::claim_offer(bob, donor_voice_address); + multi_action::claim_offer(bob, donor_voice_address); //need to be caged to finalize donor directed workflow and release control of the account multi_action::finalize_and_cage(&resource_sig, vector::length(&vals)); @@ -578,7 +578,7 @@ module ol_framework::test_donor_voice { multi_action::claim_offer(alice, donor_voice_address); multi_action::claim_offer(bob, donor_voice_address); multi_action::claim_offer(carol, donor_voice_address); - + //need to be caged to finalize donor directed workflow and release control of the account multi_action::finalize_and_cage(&resource_sig, 2); @@ -676,7 +676,7 @@ module ol_framework::test_donor_voice { // Dave and Eve are unhappy, and vote to liquidate the account. let vals = mock::genesis_n_vals(root, 5); // need to include eve to init funds - mock::ol_initialize_coin_and_fund_vals(root, 100000, true); + mock::ol_initialize_coin_and_fund_vals(root, 100_000, true); // start at epoch 1, since turnout tally needs epoch info, and 0 may cause // issues mock::trigger_epoch(root); @@ -723,7 +723,6 @@ module ol_framework::test_donor_voice { assert!(*vector::borrow(&addrs, 0) == @0x1000e, 7357006); assert!(*vector::borrow(&addrs, 1) == @0x1000d, 7357007); - let eve_donation_pro_rata = vector::borrow(&refunds, 0); let superman_3 = 1; // rounding from fixed_point32 assert!((*eve_donation_pro_rata + superman_3) == eve_donation, 7357008); @@ -732,7 +731,6 @@ module ol_framework::test_donor_voice { let superman_3 = 1; // rounding from fixed_point32 assert!((*dave_donation_pro_rata + superman_3) == dave_donation, 7357009); - let (_, program_balance_pre) = ol_account::balance(donor_voice_address); let (_, eve_balance_pre) = ol_account::balance(@0x1000e); @@ -747,10 +745,9 @@ module ol_framework::test_donor_voice { // eve shoul have received funds back assert!(eve_balance > eve_balance_pre, 7357010); - let (lifetime_burn_now, _) = burn::get_lifetime_tracker(); // nothing should have been burned, it was a refund + let (lifetime_burn_now, _) = burn::get_lifetime_tracker(); assert!(lifetime_burn_now == lifetime_burn_pre, 7357011); - } #[test(root = @ol_framework, alice = @0x1000a, dave = @0x1000d, eve = @0x1000e)] diff --git a/framework/libra-framework/sources/ol_sources/tests/jail.test.move b/framework/libra-framework/sources/ol_sources/tests/jail.test.move index 4e0457d7e..8ea8cfbc1 100644 --- a/framework/libra-framework/sources/ol_sources/tests/jail.test.move +++ b/framework/libra-framework/sources/ol_sources/tests/jail.test.move @@ -4,6 +4,7 @@ module ol_framework::test_jail { use std::vector; use ol_framework::mock; use ol_framework::jail; + use ol_framework::randomness; // use diem_std::debug::print; #[test(root = @ol_framework)] @@ -76,7 +77,7 @@ module ol_framework::test_jail { #[test(root = @ol_framework)] public entry fun sort_by_jail(root: signer) { - + randomness::initialize_for_testing(&root); let vals = mock::genesis_n_vals(&root, 5); let alice = *vector::borrow(&vals, 0); let bob = *vector::borrow(&vals, 1); diff --git a/framework/libra-framework/sources/ol_sources/tests/musical_chairs.test.move b/framework/libra-framework/sources/ol_sources/tests/musical_chairs.test.move index e6e6a485b..6305519be 100644 --- a/framework/libra-framework/sources/ol_sources/tests/musical_chairs.test.move +++ b/framework/libra-framework/sources/ol_sources/tests/musical_chairs.test.move @@ -3,61 +3,59 @@ /// tests for external apis, and where a dependency cycle with genesis is created. module ol_framework::test_musical_chairs { - use ol_framework::musical_chairs; - use ol_framework::mock; - use std::vector; - use std::fixed_point32; + use ol_framework::musical_chairs; + use ol_framework::mock; + use std::vector; + use std::fixed_point32; - // use diem_std::debug::print; + // use diem_std::debug::print; - #[test(root = @ol_framework)] - public entry fun eval_compliance_happy(root: signer) { + #[test(root = @ol_framework)] + public entry fun eval_compliance_happy(root: signer) { - let musical_chairs_default_seats = 10; - let vals = mock::genesis_n_vals(&root, 5); - assert!(vector::length(&vals) == 5, 7357001); + let musical_chairs_default_seats = 10; + let vals = mock::genesis_n_vals(&root, 5); + assert!(vector::length(&vals) == 5, 7357001); - // all vals compliant - mock::mock_all_vals_good_performance(&root); - let epoch = 10; // we don't evaluate epoch 0, 1 - let round = 2000; // we don't evaluate is rounds are below 1000 (one thousand) - let (good, bad, ratio) = musical_chairs::test_eval_compliance(&root, vals, - epoch, round); - assert!(vector::length(&good) == 5, 7357002); - assert!(vector::length(&bad) == 0, 7357003); - assert!(fixed_point32::is_zero(ratio), 7357004); + // all vals compliant + mock::mock_all_vals_good_performance(&root); + let epoch = 10; // we don't evaluate epoch 0, 1 + let round = 2000; // we don't evaluate is rounds are below 1000 (one thousand) + let (good, bad, ratio) = musical_chairs::test_eval_compliance(&root, vals, + epoch, round); + assert!(vector::length(&good) == 5, 7357002); + assert!(vector::length(&bad) == 0, 7357003); + assert!(fixed_point32::is_zero(ratio), 7357004); + let (outgoing_compliant_set, new_set_size) = + musical_chairs::test_stop(&root, epoch, round); - let (outgoing_compliant_set, new_set_size) = - musical_chairs::test_stop(&root, epoch, round); + assert!(vector::length(&outgoing_compliant_set) == 5, 7357005); + assert!(new_set_size == (musical_chairs_default_seats + 1), 7357006); + } + #[test(root = @ol_framework)] + // only one seat opens up + public entry fun eval_compliance_one_val(root: signer) { - assert!(vector::length(&outgoing_compliant_set) == 5, 7357005); - assert!(new_set_size == (musical_chairs_default_seats + 1), 7357006); - } + let vals = mock::genesis_n_vals(&root, 5); + assert!(vector::length(&vals) == 5, 7357001); - #[test(root = @ol_framework)] - // only one seat opens up - public entry fun eval_compliance_one_val(root: signer) { + // all vals compliant + mock::mock_case_1(&root, *vector::borrow(&vals, 0)); - let vals = mock::genesis_n_vals(&root, 5); - assert!(vector::length(&vals) == 5, 7357001); + let epoch = 10; // we don't evaluate epoch 0, 1 + let round = 2000; // we don't evaluate is rounds are below 1000 (one thousand) + let (good, bad, bad_ratio) = musical_chairs::test_eval_compliance(&root, + vals, epoch, round); + assert!(vector::length(&good) == 1, 7357002); + assert!(vector::length(&bad) == 4, 7357003); + assert!(!fixed_point32::is_zero(bad_ratio), 7357004); + assert!(fixed_point32::create_from_rational(4, 5) == bad_ratio, 7357005); - // all vals compliant - mock::mock_case_1(&root, *vector::borrow(&vals, 0)); - let epoch = 10; // we don't evaluate epoch 0, 1 - let round = 2000; // we don't evaluate is rounds are below 1000 (one thousand) - let (good, bad, bad_ratio) = musical_chairs::test_eval_compliance(&root, - vals, epoch, round); - assert!(vector::length(&good) == 1, 7357002); - assert!(vector::length(&bad) == 4, 7357003); - assert!(!fixed_point32::is_zero(bad_ratio), 7357004); - assert!(fixed_point32::create_from_rational(4, 5) == bad_ratio, 7357005); + let (_outgoing_compliant_set, _new_set_size) = + musical_chairs::test_stop(&root, epoch, round); - - let (_outgoing_compliant_set, _new_set_size) = - musical_chairs::test_stop(&root, epoch, round); - - } + } } diff --git a/framework/libra-framework/sources/ol_sources/tests/proof_of_fee.test.move b/framework/libra-framework/sources/ol_sources/tests/proof_of_fee.test.move index 8be5ac065..26a849121 100644 --- a/framework/libra-framework/sources/ol_sources/tests/proof_of_fee.test.move +++ b/framework/libra-framework/sources/ol_sources/tests/proof_of_fee.test.move @@ -21,8 +21,6 @@ module ol_framework::test_pof { const Eve: address = @0x1000e; const Frank: address = @0x1000f; - - #[test_only] fun mock_good_bid(_root: &signer, alice: &address) { let a_sig = account::create_signer_for_test(*alice); @@ -56,13 +54,11 @@ module ol_framework::test_pof { assert!(bid == 0, 1001); assert!(expires == 0, 1002); - proof_of_fee::pof_update_bid(&a_sig, 100, 0); let (bid, expires) = proof_of_fee::current_bid(*alice); assert!(bid == 100, 1003); assert!(expires == 0, 1004); - // now retract proof_of_fee::pof_retract_bid(a_sig); let (bid, expires) = proof_of_fee::current_bid(*alice); @@ -331,18 +327,15 @@ module ol_framework::test_pof { let sorted_two = proof_of_fee::get_bidders(true); assert!(vector::length(&sorted_two) != vector::length(&val_universe), 1004); - assert!(vector::length(&sorted_two) == vector::length(&val_universe) - 1, 1005); - } // We can send the fill seats function a list of validators, and the list of performing validators, and it will return the winning bidders and the bid. #[test(root = @ol_framework)] fun fill_seats_happy(root: signer) { let set = mock::genesis_n_vals(&root, 5); - mock::ol_initialize_coin_and_fund_vals(&root, 500000, true); let len = vector::length(&set); - + mock::ol_initialize_coin_and_fund_vals(&root, 500000, true); mock::pof_default(); slow_wallet::slow_wallet_epoch_drip(&root, 500000); @@ -351,7 +344,6 @@ module ol_framework::test_pof { assert!(vector::length(&sorted) == vector::length(&set), 1003); let (seats, _, _, _, _) = proof_of_fee::fill_seats_and_get_price(&root, len, &sorted, &sorted); - assert!(vector::contains(&seats, vector::borrow(&set, 0)), 1004); // filling the seat updated the computation of the consensu reward. @@ -359,7 +351,6 @@ module ol_framework::test_pof { assert!(reward == 1000000, 1005); assert!(clear_percent == 1, 1006); assert!(median_bid == 3, 1007); - } // We fill all the seats, and run the thermostat @@ -410,11 +401,9 @@ module ol_framework::test_pof { mock::ol_initialize_coin_and_fund_vals(&root, 500000, true); mock::pof_default(); - // Ok now EVE changes her mind. Will force the bid to expire. let a_sig = account::create_signer_for_test(*vector::borrow(&set, 4)); proof_of_fee::pof_update_bid(&a_sig, 0, 0); - slow_wallet::slow_wallet_epoch_drip(&root, 500000); let sorted = proof_of_fee::get_bidders(true); @@ -452,14 +441,9 @@ module ol_framework::test_pof { fun fill_seats_many_bidders(root: signer) { let set = mock::genesis_n_vals(&root, 5); mock::pof_default(); - mock::ol_initialize_coin_and_fund_vals(&root, 500000, true); - let sorted = proof_of_fee::get_bidders(true); - - - let set_size = 3; let (seats, _, _, _, _) = proof_of_fee::fill_seats_and_get_price(&root, set_size, &sorted, &sorted); @@ -610,4 +594,59 @@ module ol_framework::test_pof { assert!(median_bid == 3, 10014); } + // Tests for query_reward_adjustment + + #[test(root = @ol_framework)] + public fun test_query_reward_adjustment_no_change(root: &signer) { + use diem_framework::chain_id; + proof_of_fee::init_genesis_baseline_reward(root); + chain_id::initialize_for_test(root, 4); + + // 16 entries all with value 600 + let median_history = vector[600, 600, 600, 600, 600, 600, 600, 600, 600, 600, 600, 600, 600, 600, 600, 600]; + let nominal_reward = 1000; + + proof_of_fee::test_mock_reward(root, nominal_reward, 500, 500, median_history); + + let (did_run, did_increment, amount) = proof_of_fee::query_reward_adjustment(); + assert!(did_run == true, 7357043); + assert!(did_increment == false, 7357044); + assert!(amount == 0, 7357045); + } + + #[test(root = @ol_framework)] + public fun test_query_reward_adjustment_increase(root: &signer) { + use diem_framework::chain_id; + proof_of_fee::init_genesis_baseline_reward(root); + chain_id::initialize_for_test(root, 4); + + // 11 entries all with value 400 + let median_history = vector[400, 400, 400, 400, 400, 400, 400, 400, 400, 400, 400]; + let nominal_reward = 1000; + + proof_of_fee::test_mock_reward(root, nominal_reward, 500, 500, median_history); + + let (did_run, did_increment, amount) = proof_of_fee::query_reward_adjustment(); + assert!(did_run == true, 7357046); + assert!(did_increment == true, 7357047); + assert!(amount == nominal_reward / 10, 7357048); + } + + #[test(root = @ol_framework)] + public fun test_query_reward_adjustment_decrease(root: &signer) { + use diem_framework::chain_id; + proof_of_fee::init_genesis_baseline_reward(root); + chain_id::initialize_for_test(root, 4); + + // 11 entries all with value 960 + let median_history = vector[960, 960, 960, 960, 960, 960, 960, 960, 960, 960, 960]; + let nominal_reward = 1000; + + proof_of_fee::test_mock_reward(root, nominal_reward, 500, 500, median_history); + + let (did_run, did_increment, amount) = proof_of_fee::query_reward_adjustment(); + assert!(did_run == true, 7357049); + assert!(did_increment == false, 7357050); + assert!(amount == nominal_reward / 10, 7357051); + } } diff --git a/framework/libra-framework/sources/ol_sources/tests/slow_wallet.test.move b/framework/libra-framework/sources/ol_sources/tests/slow_wallet.test.move index b640e0c26..e4b9103c6 100644 --- a/framework/libra-framework/sources/ol_sources/tests/slow_wallet.test.move +++ b/framework/libra-framework/sources/ol_sources/tests/slow_wallet.test.move @@ -18,7 +18,7 @@ module ol_framework::test_slow_wallet { use std::vector; use std::signer; - // use diem_std::debug::print; + use diem_std::debug::print; #[test(root = @ol_framework)] // we are testing that genesis creates the needed struct @@ -112,8 +112,8 @@ module ol_framework::test_slow_wallet { fun test_epoch_drip(root: signer) { let set = mock::genesis_n_vals(&root, 4); mock::ol_initialize_coin_and_fund_vals(&root, 100, false); - let a = *vector::borrow(&set, 0); + let a = *vector::borrow(&set, 0); assert!(slow_wallet::is_slow(a), 7357000); assert!(slow_wallet::unlocked_amount(a) == 100, 735701); @@ -121,12 +121,13 @@ module ol_framework::test_slow_wallet { rewards::test_helper_pay_reward(&root, a, coin, 0); let (u, b) = ol_account::balance(a); - assert!(b==100000100, 735702); + print(&b); + assert!(b==500_000_100, 735702); assert!(u==100, 735703); slow_wallet::slow_wallet_epoch_drip(&root, 233); let (u, b) = ol_account::balance(a); - assert!(b==100000100, 735704); + assert!(b==500_000_100, 735704); assert!(u==333, 735705); } diff --git a/framework/libra-framework/sources/ol_sources/tests/stake.test.move b/framework/libra-framework/sources/ol_sources/tests/stake.test.move index bdb19a01c..047ec5ae8 100644 --- a/framework/libra-framework/sources/ol_sources/tests/stake.test.move +++ b/framework/libra-framework/sources/ol_sources/tests/stake.test.move @@ -7,7 +7,7 @@ module ol_framework::test_stake { use ol_framework::testnet; use ol_framework::grade; - // use diem_std::debug::print; + //use diem_std::debug::print; // Scenario: can take 6 already initialized validators, from a previous set // and reduce the set to 3 of those validators. @@ -44,6 +44,9 @@ module ol_framework::test_stake { // within these ranges we have different rules to recover. #[test(root = @ol_framework)] fun failover_scenarios(root: signer) { + use diem_framework::randomness; + randomness::initialize_for_testing(&root); + let set = mock::genesis_n_vals(&root, 10); testnet::unset(&root); // set to production mode @@ -60,7 +63,7 @@ module ol_framework::test_stake { // should return 10 let proposed = vals; let cfg_list = stake::check_failover_rules(proposed, 11); - assert!(vector::length(&cfg_list) == 10, 1003); + assert!(vector::length(&cfg_list) == 10, 1002); // second case: // number of seats has no effect, as long as proposed in above mininum for // happy case(9). @@ -76,24 +79,24 @@ module ol_framework::test_stake { // First case: We qualified fewer vals (7) than our theshhold (10). The // number of seats (7) is same as qualifying (the typical case) let cfg_list = stake::check_failover_rules(proposed, 7); - assert!(vector::length(&cfg_list) == 7, 1004); + assert!(vector::length(&cfg_list) == 7, 1005); // Second case: we have more qualified vals (7) than we intended to seat (6) // but it's all below our minimum (9). Take everyone that qualified. // This case is likely a bug let cfg_list = stake::check_failover_rules(proposed, 6); - assert!(vector::length(&cfg_list) == 7, 1004); + assert!(vector::length(&cfg_list) == 7, 1006); // C) Defibrillator // First case: we are not qualifying enough validators to stay alive // let's check we can get at least up to the number of seats offered let cfg_list = stake::check_failover_rules(proposed, 8); - assert!(vector::length(&cfg_list) == 8, 1004); + assert!(vector::length(&cfg_list) == 8, 1007); // Second case : we are not qualifying enough validators to stay alive // but the seats proposed is unrealistic. Let's just get back to a healthy let cfg_list = stake::check_failover_rules(proposed, 50); - assert!(vector::length(&cfg_list) == 9, 1004); + assert!(vector::length(&cfg_list) == 9, 1008); // even an empty list, but with seats to fill will work let cfg_list = stake::check_failover_rules(vector::empty(), 8); @@ -103,23 +106,20 @@ module ol_framework::test_stake { // health (9), even if musical chairs picked more (12). (Though this case should // never happen) let cfg_list = stake::check_failover_rules(vector::empty(), 12); - assert!(vector::length(&cfg_list) == 9, 1004); + assert!(vector::length(&cfg_list) == 9, 1009); // D) Hail Mary. If nothing we've done gets us above 4 proposed validators, // then don't change anything. let cfg_list = stake::check_failover_rules(vector::empty(), 0); - assert!(vector::length(&cfg_list) == 4, 1004); + assert!(vector::length(&cfg_list) == 4, 1010); // Other corner cases // musical chairs fails, but we have a list of qualifying validators // above threshold let cfg_list = stake::check_failover_rules(vals, 2); - assert!(vector::length(&cfg_list) == 10, 1004); - - + assert!(vector::length(&cfg_list) == 10, 1011); } - // Scenario: in production mode, we can't have fewer than 4 validators. // when that happens, the first failover is to get the validators with the most successful proposals. #[test(root = @ol_framework)] diff --git a/framework/libra-framework/sources/ol_sources/tests/validator_reward.test.move b/framework/libra-framework/sources/ol_sources/tests/validator_reward.test.move index 8d41753b5..bcccaba78 100644 --- a/framework/libra-framework/sources/ol_sources/tests/validator_reward.test.move +++ b/framework/libra-framework/sources/ol_sources/tests/validator_reward.test.move @@ -4,6 +4,7 @@ module ol_framework::test_reconfiguration { use std::vector; use diem_framework::stake; + use diem_framework::transaction_fee; // use diem_framework::coin; use ol_framework::mock; use ol_framework::testnet; @@ -12,90 +13,112 @@ module ol_framework::test_reconfiguration { use diem_framework::reconfiguration; use ol_framework::epoch_helper; use ol_framework::ol_account; + use ol_framework::infra_escrow; - // use diem_std::debug::print; + use diem_std::debug::print; // Scenario: all genesis validators make it to next epoch #[test(root = @ol_framework)] fun reconfig_reward_happy_case(root: signer) { - let vals = mock::genesis_n_vals(&root, 5); + let vals = mock::genesis_n_vals(&root, 5); + mock::pof_default(); + assert!(vector::length(&vals) == 5, 7357001); + let vals = stake::get_current_validators(); + assert!(vector::length(&vals) == 5, 7357002); + // all vals compliant + mock::mock_all_vals_good_performance(&root); + let (unlocked, alice_bal) = ol_account::balance(@0x1000a); + assert!(unlocked==0, 7367001); + assert!(alice_bal==0, 7357002); - mock::pof_default(); - assert!(vector::length(&vals) == 5, 7357001); - let vals = stake::get_current_validators(); - assert!(vector::length(&vals) == 5, 7357002); - // all vals compliant - mock::mock_all_vals_good_performance(&root); + let (reward_one, _entry_fee, _, _ ) = proof_of_fee::get_consensus_reward(); - let (unlocked, alice_bal) = ol_account::balance(@0x1000a); - assert!(unlocked==0, 7367001); - assert!(alice_bal==0, 7357002); + // The epoch's reward BEFORE reconfiguration + assert!(reward_one == 1000000, 7357004); - let (reward_one, _entry_fee, _, _ ) = proof_of_fee::get_consensus_reward(); - // The epoch's reward BEFORE reconfiguration - assert!(reward_one == 1000000, 7357004); - // run ol reconfiguration - mock::trigger_epoch(&root); + let infra = infra_escrow::infra_escrow_balance(); + print(&555); + print(&infra); - let vals = stake::get_current_validators(); + let subsidy = transaction_fee::system_fees_collected(); + print(&666); + print(&subsidy); - assert!(vector::length(&vals) == 5, 7357005); - // let alice_bal = libra_coin::balance(@0x1000a); - let (_unlocked, alice_bal) = ol_account::balance(@0x1000a); + // run ol reconfiguration + mock::trigger_epoch(&root); - let (_, entry_fee, _, _ ) = proof_of_fee::get_consensus_reward(); - // need to check that the user paid an PoF entry fee for next epoch. - // which means the balance will be the nominal reward, net of the PoF clearing price bid - assert!(alice_bal == (reward_one-entry_fee), 7357006) + let infra = infra_escrow::infra_escrow_balance(); + print(&5552); + print(&infra); + let subsidy = transaction_fee::system_fees_collected(); + print(&6662); + print(&subsidy); - } + let vals = stake::get_current_validators(); + assert!(vector::length(&vals) == 5, 7357005); + // let alice_bal = libra_coin::balance(@0x1000a); + let (_unlocked, alice_bal) = ol_account::balance(@0x1000a); - #[test(root = @ol_framework)] - fun drop_non_performing(root: signer) { - let _vals = mock::genesis_n_vals(&root, 5); - // mock::ol_initialize_coin(&root); - mock::pof_default(); - assert!(libra_coin::balance(@0x1000a) == 0, 7357000); + let (_, entry_fee, _, _ ) = proof_of_fee::get_consensus_reward(); + // need to check that the user paid an PoF entry fee for next epoch. + // which means the balance will be the nominal reward, net of the PoF clearing price bid + assert!(alice_bal == (reward_one - entry_fee), 7357006); - // NOTE: epoch 0 and 1 are a special case, we don't run performance grades on that one. Need to move two epochs ahead - reconfiguration::test_helper_increment_epoch_dont_reconfigure(1); - reconfiguration::test_helper_increment_epoch_dont_reconfigure(1); + // test new subsidy + let (reward_two, _entry_fee, _, _ ) = proof_of_fee::get_consensus_reward(); + print(&777); + print(&reward_two); + let new_budget = reward_two * 5; + print(&new_budget); - assert!(epoch_helper::get_current_epoch() == 2, 7357001); + let subsidy = transaction_fee::system_fees_collected(); + print(&888); + print(&subsidy); - let vals = stake::get_current_validators(); - assert!(vector::length(&vals) == 5, 7357002); + } - // all vals compliant - mock::mock_all_vals_good_performance(&root); + #[test(root = @ol_framework)] + fun drop_non_performing(root: signer) { + let _vals = mock::genesis_n_vals(&root, 5); + // mock::ol_initialize_coin(&root); + mock::pof_default(); + assert!(libra_coin::balance(@0x1000a) == 0, 7357000); - // make alice non performant - mock::mock_case_4(&root, *vector::borrow(&vals, 0)); + // NOTE: epoch 0 and 1 are a special case, we don't run performance grades on that one. Need to move two epochs ahead + reconfiguration::test_helper_increment_epoch_dont_reconfigure(1); + reconfiguration::test_helper_increment_epoch_dont_reconfigure(1); - let (reward, _, _, _ ) = proof_of_fee::get_consensus_reward(); + assert!(epoch_helper::get_current_epoch() == 2, 7357001); - // run ol reconfiguration - mock::trigger_epoch(&root); - // mock::trigger_epoch(&root); + let vals = stake::get_current_validators(); + assert!(vector::length(&vals) == 5, 7357002); - let vals = stake::get_current_validators(); + // all vals compliant + mock::mock_all_vals_good_performance(&root); + + // make alice non performant + mock::mock_case_4(&root, *vector::borrow(&vals, 0)); - // one validator missing. - assert!(vector::length(&vals) == 4, 7357003); - assert!(!vector::contains(&vals, &@0x1000a), 7357004); + let (reward, _, _, _ ) = proof_of_fee::get_consensus_reward(); - let (_, entry_fee, _, _ ) = proof_of_fee::get_consensus_reward(); + // run ol reconfiguration + mock::trigger_epoch(&root); - // alice doesn't get paid + let vals = stake::get_current_validators(); - assert!(libra_coin::balance(@0x1000a) == 0, 7357005); - // bob does - assert!(libra_coin::balance(@0x1000b) == (reward - entry_fee), 7357006); + // one validator missing. + assert!(vector::length(&vals) == 4, 7357003); + assert!(!vector::contains(&vals, &@0x1000a), 7357004); + let (_, entry_fee, _, _ ) = proof_of_fee::get_consensus_reward(); + // alice doesn't get paid + assert!(libra_coin::balance(@0x1000a) == 0, 7357005); + // bob does + assert!(libra_coin::balance(@0x1000b) == (reward - entry_fee), 7357006); } diff --git a/framework/libra-framework/sources/randomness.move b/framework/libra-framework/sources/randomness.move index 53061cce5..7fab83a78 100644 --- a/framework/libra-framework/sources/randomness.move +++ b/framework/libra-framework/sources/randomness.move @@ -342,7 +342,7 @@ module diem_framework::randomness { /// Generate a permutation of `[0, 1, ..., n-1]` uniformly at random. /// If n is 0, returns the empty vector. - fun permutation(n: u64): vector acquires PerBlockRandomness { + public fun permutation(n: u64): vector acquires PerBlockRandomness { let values = vector[]; if(n == 0) { diff --git a/framework/libra-framework/sources/stake.move b/framework/libra-framework/sources/stake.move index 4d1c77dfd..d98825ebf 100644 --- a/framework/libra-framework/sources/stake.move +++ b/framework/libra-framework/sources/stake.move @@ -15,6 +15,7 @@ module diem_framework::stake { use diem_framework::system_addresses; use ol_framework::slow_wallet; use ol_framework::testnet; + use ol_framework::address_utils; // use diem_std::debug::print; @@ -878,7 +879,7 @@ module diem_framework::stake { current_vals } - /// Bubble sort the validators by their proposal counts. + /// Sort the validators by their proposal counts. public fun get_sorted_vals_by_net_props(): vector
acquires ValidatorSet, ValidatorConfig, ValidatorPerformance { let eligible_validators = get_current_validators(); let length = vector::length
(&eligible_validators); @@ -905,36 +906,12 @@ module diem_framework::stake { }; // Sorting the accounts vector based on value (weights). - // Bubble sort algorithm - let len_filtered = vector::length
(&filtered_vals); - if (len_filtered < 2) return filtered_vals; - let i = 0; - while (i < len_filtered){ - let j = 0; - while(j < len_filtered-i-1){ - - let value_j = *(vector::borrow(&weights, j)); - - let value_jp1 = *(vector::borrow(&weights, j+1)); - if(value_j > value_jp1){ - - vector::swap(&mut weights, j, j+1); - - vector::swap
(&mut filtered_vals, j, j+1); - }; - j = j + 1; - - }; - i = i + 1; - - }; - + address_utils::sort_by_values(&mut filtered_vals, &mut weights); // Reverse to have sorted order - high to low. vector::reverse
(&mut filtered_vals); return filtered_vals - } //////// 0L //////// diff --git a/framework/releases/head.mrb b/framework/releases/head.mrb index b63243772b0f84f612318448d21b2e022d261b20..14d5715b6d642cdb7a4b6128a30d563883a4fc88 100644 GIT binary patch delta 103672 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%0DaKc| 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 MeAzmz%nZQ(KSBH^rvLx| delta 88460 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

CVZ<1}E@r}IRNDuV0b=&lLF`L1*$lO# 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 -> anyhow::Result<()> { .await? .json::>() .await?; + // Find the latest version by parsing version numbers and sorting let latest_version = resp .iter() @@ -202,12 +203,18 @@ pub async fn download_genesis(home_dir: Option) -> anyhow::Result<()> { "https://raw.githubusercontent.com/0LNetworkCommunity/epoch-archive-mainnet/main/upgrades", latest_version.unwrap_or(DEFAULT_WAYPOINT_VERSION) ); + // Fetch the latest waypoint let blob_bytes = reqwest::get(&latest_path).await?.bytes().await?; let home = home_dir.unwrap_or_else(libra_types::global_config_dir); let genesis_dir = home.join("genesis/"); + + // Ensure the genesis directory exists + std::fs::create_dir_all(&genesis_dir)?; + let p = genesis_dir.join("genesis.blob"); + // Write the genesis blob to the file std::fs::write(p, &blob_bytes)?; Ok(()) } @@ -289,11 +296,28 @@ async fn persist_genesis() { let path = p.path().to_owned(); - download_genesis(Some(path)).await.unwrap(); - let l = std::fs::read_dir(p.path()) - .unwrap() - .next() - .unwrap() - .unwrap(); - assert!(l.file_name().to_str().unwrap().contains("genesis.blob")); + // Ensure the directory exists + assert!( + std::fs::metadata(&path).is_ok(), + "Directory does not exist: {:?}", + path + ); + + // Verify path is a directory + assert!( + Path::new(&path).is_dir(), + "Path is not a directory: {:?}", + path + ); + + // Attempt to download genesis + download_genesis(Some(path.clone())).await.unwrap(); + + // Check if the genesis.blob file exists + let genesis_blob_path = path.join("genesis").join("genesis.blob"); + assert!( + genesis_blob_path.exists(), + "genesis.blob file does not exist: {:?}", + genesis_blob_path + ); }