2015-05-27 12:08:46 +00:00
|
|
|
// Copyright (c) 2012-2015, The CryptoNote developers, The Bytecoin developers
|
2014-08-13 10:38:35 +00:00
|
|
|
//
|
|
|
|
// This file is part of Bytecoin.
|
|
|
|
//
|
|
|
|
// Bytecoin is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU Lesser General Public License as published by
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
// (at your option) any later version.
|
|
|
|
//
|
|
|
|
// Bytecoin is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU Lesser General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU Lesser General Public License
|
|
|
|
// along with Bytecoin. If not, see <http://www.gnu.org/licenses/>.
|
2014-03-03 22:07:58 +00:00
|
|
|
|
|
|
|
#pragma once
|
2015-07-30 15:22:07 +00:00
|
|
|
#include "Chaingen.h"
|
2014-08-13 10:51:37 +00:00
|
|
|
#include "TransactionBuilder.h"
|
2014-03-03 22:07:58 +00:00
|
|
|
|
|
|
|
const size_t invalid_index_value = std::numeric_limits<size_t>::max();
|
|
|
|
|
|
|
|
|
|
|
|
template<class concrete_test>
|
|
|
|
class gen_double_spend_base : public test_chain_unit_base
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
static const uint64_t send_amount = MK_COINS(17);
|
|
|
|
|
|
|
|
gen_double_spend_base();
|
|
|
|
|
2015-05-27 12:08:46 +00:00
|
|
|
bool check_tx_verification_context(const CryptoNote::tx_verification_context& tvc, bool tx_added, size_t event_idx, const CryptoNote::Transaction& tx);
|
|
|
|
bool check_block_verification_context(const CryptoNote::block_verification_context& bvc, size_t event_idx, const CryptoNote::Block& block);
|
2014-03-03 22:07:58 +00:00
|
|
|
|
2015-05-27 12:08:46 +00:00
|
|
|
bool mark_last_valid_block(CryptoNote::core& c, size_t ev_index, const std::vector<test_event_entry>& events);
|
|
|
|
bool mark_invalid_tx(CryptoNote::core& c, size_t ev_index, const std::vector<test_event_entry>& events);
|
|
|
|
bool mark_invalid_block(CryptoNote::core& c, size_t ev_index, const std::vector<test_event_entry>& events);
|
|
|
|
bool check_double_spend(CryptoNote::core& c, size_t ev_index, const std::vector<test_event_entry>& events);
|
2014-03-03 22:07:58 +00:00
|
|
|
|
|
|
|
private:
|
2015-05-27 12:08:46 +00:00
|
|
|
CryptoNote::Block m_last_valid_block;
|
2014-03-03 22:07:58 +00:00
|
|
|
size_t m_invalid_tx_index;
|
|
|
|
size_t m_invalid_block_index;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
template<bool txs_keeped_by_block>
|
|
|
|
struct gen_double_spend_in_tx : public gen_double_spend_base< gen_double_spend_in_tx<txs_keeped_by_block> >
|
|
|
|
{
|
|
|
|
static const uint64_t send_amount = MK_COINS(17);
|
|
|
|
static const bool has_invalid_tx = true;
|
|
|
|
static const size_t expected_pool_txs_count = 0;
|
|
|
|
static const uint64_t expected_bob_balance = send_amount;
|
|
|
|
static const uint64_t expected_alice_balance = 0;
|
|
|
|
|
|
|
|
bool generate(std::vector<test_event_entry>& events) const;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
template<bool txs_keeped_by_block>
|
|
|
|
struct gen_double_spend_in_the_same_block : public gen_double_spend_base< gen_double_spend_in_the_same_block<txs_keeped_by_block> >
|
|
|
|
{
|
|
|
|
static const uint64_t send_amount = MK_COINS(17);
|
|
|
|
static const bool has_invalid_tx = !txs_keeped_by_block;
|
|
|
|
static const size_t expected_pool_txs_count = has_invalid_tx ? 1 : 2;
|
|
|
|
static const uint64_t expected_bob_balance = send_amount;
|
|
|
|
static const uint64_t expected_alice_balance = 0;
|
|
|
|
|
|
|
|
bool generate(std::vector<test_event_entry>& events) const;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
template<bool txs_keeped_by_block>
|
|
|
|
struct gen_double_spend_in_different_blocks : public gen_double_spend_base< gen_double_spend_in_different_blocks<txs_keeped_by_block> >
|
|
|
|
{
|
|
|
|
static const uint64_t send_amount = MK_COINS(17);
|
|
|
|
static const bool has_invalid_tx = !txs_keeped_by_block;
|
|
|
|
static const size_t expected_pool_txs_count = has_invalid_tx ? 0 : 1;
|
|
|
|
static const uint64_t expected_bob_balance = 0;
|
2014-08-13 10:51:37 +00:00
|
|
|
static uint64_t expected_alice_balance;
|
|
|
|
|
|
|
|
gen_double_spend_in_different_blocks() :
|
|
|
|
gen_double_spend_base< gen_double_spend_in_different_blocks<txs_keeped_by_block> >() {
|
|
|
|
expected_alice_balance = send_amount - this->m_currency.minimumFee();
|
|
|
|
}
|
2014-03-03 22:07:58 +00:00
|
|
|
|
|
|
|
bool generate(std::vector<test_event_entry>& events) const;
|
|
|
|
};
|
|
|
|
|
2014-08-13 10:51:37 +00:00
|
|
|
template<bool txs_keeped_by_block>
|
|
|
|
uint64_t gen_double_spend_in_different_blocks<txs_keeped_by_block>::expected_alice_balance;
|
|
|
|
|
2014-03-03 22:07:58 +00:00
|
|
|
|
|
|
|
template<bool txs_keeped_by_block>
|
|
|
|
struct gen_double_spend_in_alt_chain_in_the_same_block : public gen_double_spend_base< gen_double_spend_in_alt_chain_in_the_same_block<txs_keeped_by_block> >
|
|
|
|
{
|
|
|
|
static const uint64_t send_amount = MK_COINS(17);
|
|
|
|
static const bool has_invalid_tx = !txs_keeped_by_block;
|
|
|
|
static const size_t expected_pool_txs_count = has_invalid_tx ? 1 : 2;
|
|
|
|
static const uint64_t expected_bob_balance = send_amount;
|
|
|
|
static const uint64_t expected_alice_balance = 0;
|
|
|
|
|
|
|
|
bool generate(std::vector<test_event_entry>& events) const;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
template<bool txs_keeped_by_block>
|
|
|
|
struct gen_double_spend_in_alt_chain_in_different_blocks : public gen_double_spend_base< gen_double_spend_in_alt_chain_in_different_blocks<txs_keeped_by_block> >
|
|
|
|
{
|
|
|
|
static const uint64_t send_amount = MK_COINS(17);
|
|
|
|
static const bool has_invalid_tx = !txs_keeped_by_block;
|
|
|
|
static const size_t expected_pool_txs_count = has_invalid_tx ? 1 : 2;
|
|
|
|
static const uint64_t expected_bob_balance = send_amount;
|
|
|
|
static const uint64_t expected_alice_balance = 0;
|
|
|
|
|
|
|
|
bool generate(std::vector<test_event_entry>& events) const;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class gen_double_spend_in_different_chains : public test_chain_unit_base
|
|
|
|
{
|
|
|
|
public:
|
2014-06-25 17:21:42 +00:00
|
|
|
static const uint64_t send_amount = MK_COINS(31);
|
2014-08-13 10:51:37 +00:00
|
|
|
size_t expected_blockchain_height;
|
2014-03-03 22:07:58 +00:00
|
|
|
|
|
|
|
gen_double_spend_in_different_chains();
|
|
|
|
|
|
|
|
bool generate(std::vector<test_event_entry>& events) const;
|
|
|
|
|
2015-05-27 12:08:46 +00:00
|
|
|
bool check_double_spend(CryptoNote::core& c, size_t ev_index, const std::vector<test_event_entry>& events);
|
2014-03-03 22:07:58 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2014-08-13 10:51:37 +00:00
|
|
|
class TestGenerator;
|
|
|
|
|
|
|
|
class DoubleSpendBase : public test_chain_unit_base
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
|
|
|
// parameters to be checked
|
|
|
|
uint64_t send_amount;
|
|
|
|
bool has_invalid_tx;
|
|
|
|
|
|
|
|
DoubleSpendBase();
|
|
|
|
|
2015-05-27 12:08:46 +00:00
|
|
|
bool check_tx_verification_context(const CryptoNote::tx_verification_context& tvc, bool tx_added, size_t event_idx, const CryptoNote::Transaction& tx);
|
|
|
|
bool check_block_verification_context(const CryptoNote::block_verification_context& bvc, size_t event_idx, const CryptoNote::Block& block);
|
2014-08-13 10:51:37 +00:00
|
|
|
|
2015-05-27 12:08:46 +00:00
|
|
|
bool mark_last_valid_block(CryptoNote::core& c, size_t ev_index, const std::vector<test_event_entry>& events);
|
|
|
|
bool mark_invalid_tx(CryptoNote::core& c, size_t ev_index, const std::vector<test_event_entry>& events);
|
|
|
|
bool mark_invalid_block(CryptoNote::core& c, size_t ev_index, const std::vector<test_event_entry>& events);
|
|
|
|
bool check_double_spend(CryptoNote::core& c, size_t ev_index, const std::vector<test_event_entry>& events);
|
2014-08-13 10:51:37 +00:00
|
|
|
|
|
|
|
TestGenerator prepare(std::vector<test_event_entry>& events) const;
|
|
|
|
TransactionBuilder createBobToAliceTx() const;
|
2015-04-06 16:13:07 +00:00
|
|
|
TransactionBuilder::MultisignatureSource createSource() const;
|
2014-08-13 10:51:37 +00:00
|
|
|
|
|
|
|
protected:
|
|
|
|
|
2015-07-30 15:22:07 +00:00
|
|
|
CryptoNote::AccountBase m_bob_account;
|
|
|
|
CryptoNote::AccountBase m_alice_account;
|
2015-05-27 12:08:46 +00:00
|
|
|
CryptoNote::KeyPair m_outputTxKey;
|
2014-08-13 10:51:37 +00:00
|
|
|
|
|
|
|
private:
|
|
|
|
|
2015-07-30 15:22:07 +00:00
|
|
|
Crypto::Hash m_last_valid_block;
|
2014-08-13 10:51:37 +00:00
|
|
|
size_t m_invalid_tx_index;
|
|
|
|
size_t m_invalid_block_index;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
struct MultiSigTx_DoubleSpendInTx : public DoubleSpendBase
|
|
|
|
{
|
|
|
|
const bool m_txsKeepedByBlock;
|
|
|
|
|
|
|
|
MultiSigTx_DoubleSpendInTx(bool txsKeepedByBlock);
|
|
|
|
|
|
|
|
bool generate(std::vector<test_event_entry>& events) const;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct MultiSigTx_DoubleSpendSameBlock : public DoubleSpendBase
|
|
|
|
{
|
|
|
|
const bool m_txsKeepedByBlock;
|
|
|
|
|
|
|
|
MultiSigTx_DoubleSpendSameBlock(bool txsKeepedByBlock);
|
|
|
|
|
|
|
|
bool generate(std::vector<test_event_entry>& events) const;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
struct MultiSigTx_DoubleSpendDifferentBlocks : public DoubleSpendBase
|
|
|
|
{
|
|
|
|
const bool m_txsKeepedByBlock;
|
|
|
|
|
|
|
|
MultiSigTx_DoubleSpendDifferentBlocks(bool txsKeepedByBlock);
|
|
|
|
|
|
|
|
bool generate(std::vector<test_event_entry>& events) const;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct MultiSigTx_DoubleSpendAltChainSameBlock : public DoubleSpendBase
|
|
|
|
{
|
|
|
|
const bool m_txsKeepedByBlock;
|
|
|
|
|
|
|
|
MultiSigTx_DoubleSpendAltChainSameBlock(bool txsKeepedByBlock);
|
|
|
|
|
2015-05-27 12:08:46 +00:00
|
|
|
bool check_tx_verification_context(const CryptoNote::tx_verification_context& tvc, bool tx_added, size_t event_idx, const CryptoNote::Transaction& tx) {
|
2014-08-13 10:51:37 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-05-27 12:08:46 +00:00
|
|
|
bool check_block_verification_context(const CryptoNote::block_verification_context& bvc, size_t event_idx, const CryptoNote::Block& block) {
|
2014-08-13 10:51:37 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool generate(std::vector<test_event_entry>& events) const;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct MultiSigTx_DoubleSpendAltChainDifferentBlocks : public DoubleSpendBase
|
|
|
|
{
|
|
|
|
const bool m_txsKeepedByBlock;
|
|
|
|
MultiSigTx_DoubleSpendAltChainDifferentBlocks(bool txsKeepedByBlock);
|
|
|
|
bool generate(std::vector<test_event_entry>& events) const;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2014-03-03 22:07:58 +00:00
|
|
|
#define INIT_DOUBLE_SPEND_TEST() \
|
|
|
|
uint64_t ts_start = 1338224400; \
|
|
|
|
GENERATE_ACCOUNT(miner_account); \
|
|
|
|
MAKE_GENESIS_BLOCK(events, blk_0, miner_account, ts_start); \
|
|
|
|
MAKE_ACCOUNT(events, bob_account); \
|
|
|
|
MAKE_ACCOUNT(events, alice_account); \
|
|
|
|
REWIND_BLOCKS(events, blk_0r, blk_0, miner_account); \
|
|
|
|
MAKE_TX(events, tx_0, miner_account, bob_account, send_amount, blk_0); \
|
|
|
|
MAKE_NEXT_BLOCK_TX1(events, blk_1, blk_0r, miner_account, tx_0); \
|
|
|
|
REWIND_BLOCKS(events, blk_1r, blk_1, miner_account);
|
|
|
|
|
|
|
|
|
|
|
|
#include "double_spend.inl"
|