gamma picker: relax constructor requirements and test for selecting outputs from first spendable block
This commit is contained in:
parent
44ac52f929
commit
eb0e4052b7
|
@ -1001,7 +1001,7 @@ gamma_picker::gamma_picker(const std::vector<uint64_t> &rct_offsets, double shap
|
||||||
rct_offsets(rct_offsets)
|
rct_offsets(rct_offsets)
|
||||||
{
|
{
|
||||||
gamma = std::gamma_distribution<double>(shape, scale);
|
gamma = std::gamma_distribution<double>(shape, scale);
|
||||||
THROW_WALLET_EXCEPTION_IF(rct_offsets.size() <= CRYPTONOTE_DEFAULT_TX_SPENDABLE_AGE, error::wallet_internal_error, "Bad offset calculation");
|
THROW_WALLET_EXCEPTION_IF(rct_offsets.size() < std::max(1, CRYPTONOTE_DEFAULT_TX_SPENDABLE_AGE), error::wallet_internal_error, "Bad offset calculation");
|
||||||
const size_t blocks_in_a_year = 86400 * 365 / DIFFICULTY_TARGET_V2;
|
const size_t blocks_in_a_year = 86400 * 365 / DIFFICULTY_TARGET_V2;
|
||||||
const size_t blocks_to_consider = std::min<size_t>(rct_offsets.size(), blocks_in_a_year);
|
const size_t blocks_to_consider = std::min<size_t>(rct_offsets.size(), blocks_in_a_year);
|
||||||
const size_t outputs_to_consider = rct_offsets.back() - (blocks_to_consider < rct_offsets.size() ? rct_offsets[rct_offsets.size() - blocks_to_consider - 1] : 0);
|
const size_t outputs_to_consider = rct_offsets.back() - (blocks_to_consider < rct_offsets.size() ? rct_offsets[rct_offsets.size() - blocks_to_consider - 1] : 0);
|
||||||
|
@ -8420,7 +8420,7 @@ void wallet2::get_outs(std::vector<std::vector<tools::wallet2::get_outs_entry>>
|
||||||
if (has_rct)
|
if (has_rct)
|
||||||
{
|
{
|
||||||
// check we're clear enough of rct start, to avoid corner cases below
|
// check we're clear enough of rct start, to avoid corner cases below
|
||||||
THROW_WALLET_EXCEPTION_IF(rct_offsets.size() <= CRYPTONOTE_DEFAULT_TX_SPENDABLE_AGE,
|
THROW_WALLET_EXCEPTION_IF(rct_offsets.size() < std::max(1, CRYPTONOTE_DEFAULT_TX_SPENDABLE_AGE),
|
||||||
error::get_output_distribution, "Not enough rct outputs");
|
error::get_output_distribution, "Not enough rct outputs");
|
||||||
THROW_WALLET_EXCEPTION_IF(rct_offsets.back() <= max_rct_index,
|
THROW_WALLET_EXCEPTION_IF(rct_offsets.back() <= max_rct_index,
|
||||||
error::get_output_distribution, "Daemon reports suspicious number of rct outputs");
|
error::get_output_distribution, "Daemon reports suspicious number of rct outputs");
|
||||||
|
|
|
@ -218,3 +218,59 @@ TEST(select_outputs, same_distribution)
|
||||||
MDEBUG("avg_dev: " << avg_dev);
|
MDEBUG("avg_dev: " << avg_dev);
|
||||||
ASSERT_LT(avg_dev, 0.02);
|
ASSERT_LT(avg_dev, 0.02);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TEST(select_outputs, exact_unlock_block)
|
||||||
|
{
|
||||||
|
std::vector<uint64_t> offsets;
|
||||||
|
MKOFFSETS(300000, 1 + (crypto::rand<size_t>() & 0x1f));
|
||||||
|
tools::gamma_picker picker(offsets);
|
||||||
|
|
||||||
|
// Calculate output offset ranges for the very first block that is spendable.
|
||||||
|
// Since gamma_picker is passed data for EXISTING blocks. The last block it can select outputs
|
||||||
|
// from *inclusive* that is allowed by consensus is the
|
||||||
|
// -(CRYPTONOTE_DEFAULT_TX_SPENDABLE_AGE - 1)th block passed to the gamma picker.
|
||||||
|
// In the case that there is not unlock time limit enforced by the protocol, this pointer would
|
||||||
|
// point to rct_offsets.end() [the address of the element after the last existing element]
|
||||||
|
const uint64_t* const first_block_too_young = offsets.data() + offsets.size() - (std::max(CRYPTONOTE_DEFAULT_TX_SPENDABLE_AGE, 1) - 1);
|
||||||
|
const uint64_t exact_block_offsets_start_inclusive = *(first_block_too_young - 2);
|
||||||
|
const uint64_t exact_block_offsets_stop_exclusive = *(first_block_too_young - 1);
|
||||||
|
|
||||||
|
// if too low we may fail by not picking exact block
|
||||||
|
// if too high test is not as senstive as it could be
|
||||||
|
constexpr size_t NUM_PICK_TESTS = 1 << 20;
|
||||||
|
|
||||||
|
bool picked_exact_unlock_block = false;
|
||||||
|
for (size_t i = 0; i < NUM_PICK_TESTS; ++i)
|
||||||
|
{
|
||||||
|
const uint64_t picked_i = picker.pick();
|
||||||
|
if (picked_i >= n_outs) // routine bad pick, handle by looping
|
||||||
|
continue;
|
||||||
|
|
||||||
|
ASSERT_LT(picked_i, exact_block_offsets_stop_exclusive); // This pick is too young
|
||||||
|
|
||||||
|
if (picked_i >= exact_block_offsets_start_inclusive)
|
||||||
|
{
|
||||||
|
// this pick is for the youngest valid spendable block
|
||||||
|
picked_exact_unlock_block = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
EXPECT_TRUE(picked_exact_unlock_block);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(select_outputs, exact_unlock_block_tiny)
|
||||||
|
{
|
||||||
|
// Create chain of length CRYPTONOTE_DEFAULT_TX_SPENDABLE_AGE where there is one output in block 0
|
||||||
|
std::vector<uint64_t> offsets(std::max(CRYPTONOTE_DEFAULT_TX_SPENDABLE_AGE, 1), 0);
|
||||||
|
offsets[0] = 1;
|
||||||
|
tools::gamma_picker picker(offsets);
|
||||||
|
|
||||||
|
constexpr size_t MAX_PICK_TRIES = 10;
|
||||||
|
bool found_the_one_output = false;
|
||||||
|
for (size_t i = 0; i < MAX_PICK_TRIES; ++i)
|
||||||
|
if (picker.pick() == 0)
|
||||||
|
found_the_one_output = true;
|
||||||
|
|
||||||
|
EXPECT_TRUE(found_the_one_output);
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in New Issue