PageRenderTime 64ms CodeModel.GetById 22ms RepoModel.GetById 0ms app.codeStats 0ms

/tests/core_tests/ring_signature_1.cpp

https://gitlab.com/nexxuz/cryptonote
C++ | 318 lines | 235 code | 59 blank | 24 comment | 6 complexity | 422738abac3080f5c78871d6d4488e97 MD5 | raw file
  1. // Copyright (c) 2011-2014 The Cryptonote developers
  2. // Distributed under the MIT/X11 software license, see the accompanying
  3. // file COPYING or http://www.opensource.org/licenses/mit-license.php.
  4. #include "chaingen.h"
  5. #include "chaingen_tests_list.h"
  6. using namespace epee;
  7. using namespace cryptonote;
  8. ////////
  9. // class gen_ring_signature_1;
  10. gen_ring_signature_1::gen_ring_signature_1()
  11. {
  12. REGISTER_CALLBACK("check_balances_1", gen_ring_signature_1::check_balances_1);
  13. REGISTER_CALLBACK("check_balances_2", gen_ring_signature_1::check_balances_2);
  14. }
  15. namespace
  16. {
  17. // To be sure that miner tx outputs don't match any bob_account and some_accounts inputs
  18. const uint64_t rnd_11 = 475921;
  19. const uint64_t rnd_20 = 360934;
  20. const uint64_t rnd_29 = 799665;
  21. }
  22. bool gen_ring_signature_1::generate(std::vector<test_event_entry>& events) const
  23. {
  24. uint64_t ts_start = 1338224400;
  25. GENERATE_ACCOUNT(miner_account);
  26. // events
  27. MAKE_GENESIS_BLOCK(events, blk_0, miner_account, ts_start); // 0
  28. MAKE_ACCOUNT(events, some_account_1); // 1
  29. MAKE_ACCOUNT(events, some_account_2); // 2
  30. MAKE_ACCOUNT(events, bob_account); // 3
  31. MAKE_ACCOUNT(events, alice_account); // 4
  32. MAKE_NEXT_BLOCK(events, blk_1, blk_0, miner_account); // 5
  33. MAKE_NEXT_BLOCK(events, blk_2, blk_1, miner_account); // 6
  34. MAKE_NEXT_BLOCK(events, blk_3, blk_2, miner_account); // 7
  35. MAKE_NEXT_BLOCK(events, blk_4, blk_3, miner_account); // 8
  36. REWIND_BLOCKS(events, blk_5, blk_4, miner_account); // <N blocks>
  37. REWIND_BLOCKS(events, blk_5r, blk_5, miner_account); // <N blocks>
  38. MAKE_TX_LIST_START(events, txs_blk_6, miner_account, bob_account, MK_COINS(1), blk_5); // 9 + 2N
  39. MAKE_TX_LIST(events, txs_blk_6, miner_account, bob_account, MK_COINS(11) + rnd_11, blk_5); // 10 + 2N
  40. MAKE_TX_LIST(events, txs_blk_6, miner_account, bob_account, MK_COINS(11) + rnd_11, blk_5); // 11 + 2N
  41. MAKE_TX_LIST(events, txs_blk_6, miner_account, bob_account, MK_COINS(20) + rnd_20, blk_5); // 12 + 2N
  42. MAKE_TX_LIST(events, txs_blk_6, miner_account, bob_account, MK_COINS(29) + rnd_29, blk_5); // 13 + 2N
  43. MAKE_TX_LIST(events, txs_blk_6, miner_account, bob_account, MK_COINS(29) + rnd_29, blk_5); // 14 + 2N
  44. MAKE_TX_LIST(events, txs_blk_6, miner_account, bob_account, MK_COINS(29) + rnd_29, blk_5); // 15 + 2N
  45. MAKE_TX_LIST(events, txs_blk_6, miner_account, some_account_1, MK_COINS(11) + rnd_11, blk_5); // 16 + 2N
  46. MAKE_TX_LIST(events, txs_blk_6, miner_account, some_account_1, MK_COINS(11) + rnd_11, blk_5); // 17 + 2N
  47. MAKE_TX_LIST(events, txs_blk_6, miner_account, some_account_1, MK_COINS(11) + rnd_11, blk_5); // 18 + 2N
  48. MAKE_TX_LIST(events, txs_blk_6, miner_account, some_account_1, MK_COINS(11) + rnd_11, blk_5); // 19 + 2N
  49. MAKE_TX_LIST(events, txs_blk_6, miner_account, some_account_1, MK_COINS(20) + rnd_20, blk_5); // 20 + 2N
  50. MAKE_TX_LIST(events, txs_blk_6, miner_account, some_account_2, MK_COINS(20) + rnd_20, blk_5); // 21 + 2N
  51. MAKE_NEXT_BLOCK_TX_LIST(events, blk_6, blk_5r, miner_account, txs_blk_6); // 22 + 2N
  52. DO_CALLBACK(events, "check_balances_1"); // 23 + 2N
  53. REWIND_BLOCKS(events, blk_6r, blk_6, miner_account); // <N blocks>
  54. // 129 = 11 + 11 + 20 + 29 + 29 + 29
  55. MAKE_TX_MIX(events, tx_0, bob_account, alice_account, MK_COINS(129) + 2 * rnd_11 + rnd_20 + 3 * rnd_29 - TESTS_DEFAULT_FEE, 2, blk_6); // 24 + 3N
  56. MAKE_NEXT_BLOCK_TX1(events, blk_7, blk_6r, miner_account, tx_0); // 25 + 3N
  57. DO_CALLBACK(events, "check_balances_2"); // 26 + 3N
  58. return true;
  59. }
  60. bool gen_ring_signature_1::check_balances_1(cryptonote::core& c, size_t ev_index, const std::vector<test_event_entry>& events)
  61. {
  62. DEFINE_TESTS_ERROR_CONTEXT("gen_ring_signature_1::check_balances_1");
  63. m_bob_account = boost::get<account_base>(events[3]);
  64. m_alice_account = boost::get<account_base>(events[4]);
  65. std::list<block> blocks;
  66. bool r = c.get_blocks(0, 100 + 2 * CRYPTONOTE_MINED_MONEY_UNLOCK_WINDOW, blocks);
  67. CHECK_TEST_CONDITION(r);
  68. std::vector<cryptonote::block> chain;
  69. map_hash2tx_t mtx;
  70. r = find_block_chain(events, chain, mtx, get_block_hash(blocks.back()));
  71. CHECK_TEST_CONDITION(r);
  72. CHECK_EQ(MK_COINS(130) + 2 * rnd_11 + rnd_20 + 3 * rnd_29, get_balance(m_bob_account, chain, mtx));
  73. CHECK_EQ(0, get_balance(m_alice_account, chain, mtx));
  74. return true;
  75. }
  76. bool gen_ring_signature_1::check_balances_2(cryptonote::core& c, size_t ev_index, const std::vector<test_event_entry>& events)
  77. {
  78. DEFINE_TESTS_ERROR_CONTEXT("gen_ring_signature_1::check_balances_2");
  79. std::list<block> blocks;
  80. bool r = c.get_blocks(0, 100 + 2 * CRYPTONOTE_MINED_MONEY_UNLOCK_WINDOW, blocks);
  81. CHECK_TEST_CONDITION(r);
  82. std::vector<cryptonote::block> chain;
  83. map_hash2tx_t mtx;
  84. r = find_block_chain(events, chain, mtx, get_block_hash(blocks.back()));
  85. CHECK_TEST_CONDITION(r);
  86. CHECK_EQ(MK_COINS(1), get_balance(m_bob_account, chain, mtx));
  87. CHECK_EQ(MK_COINS(129) + 2 * rnd_11 + rnd_20 + 3 * rnd_29 - TESTS_DEFAULT_FEE, get_balance(m_alice_account, chain, mtx));
  88. return true;
  89. }
  90. ////////
  91. // class gen_ring_signature_2;
  92. gen_ring_signature_2::gen_ring_signature_2()
  93. {
  94. REGISTER_CALLBACK("check_balances_1", gen_ring_signature_2::check_balances_1);
  95. REGISTER_CALLBACK("check_balances_2", gen_ring_signature_2::check_balances_2);
  96. }
  97. /**
  98. * Bob has 4 inputs by 61 coins. He sends 4 * 61 coins to Alice, using ring signature with nmix = 3. Each Bob's input
  99. * is used as mix for 3 others.
  100. */
  101. bool gen_ring_signature_2::generate(std::vector<test_event_entry>& events) const
  102. {
  103. uint64_t ts_start = 1338224400;
  104. GENERATE_ACCOUNT(miner_account);
  105. // events
  106. MAKE_GENESIS_BLOCK(events, blk_0, miner_account, ts_start); // 0
  107. MAKE_ACCOUNT(events, bob_account); // 1
  108. MAKE_ACCOUNT(events, alice_account); // 2
  109. MAKE_NEXT_BLOCK(events, blk_1, blk_0, miner_account); // 3
  110. MAKE_NEXT_BLOCK(events, blk_2, blk_1, miner_account); // 4
  111. MAKE_NEXT_BLOCK(events, blk_3, blk_2, miner_account); // 5
  112. REWIND_BLOCKS(events, blk_3r, blk_3, miner_account); // <N blocks>
  113. MAKE_TX_LIST_START(events, txs_blk_4, miner_account, bob_account, MK_COINS(61), blk_3); // 6 + N
  114. MAKE_TX_LIST(events, txs_blk_4, miner_account, bob_account, MK_COINS(61), blk_3); // 7 + N
  115. MAKE_TX_LIST(events, txs_blk_4, miner_account, bob_account, MK_COINS(61), blk_3); // 8 + N
  116. MAKE_TX_LIST(events, txs_blk_4, miner_account, bob_account, MK_COINS(61), blk_3); // 9 + N
  117. MAKE_NEXT_BLOCK_TX_LIST(events, blk_4, blk_3r, miner_account, txs_blk_4); // 10 + N
  118. DO_CALLBACK(events, "check_balances_1"); // 11 + N
  119. REWIND_BLOCKS(events, blk_4r, blk_4, miner_account); // <N blocks>
  120. MAKE_TX_MIX(events, tx_0, bob_account, alice_account, MK_COINS(244) - TESTS_DEFAULT_FEE, 3, blk_4); // 12 + 2N
  121. MAKE_NEXT_BLOCK_TX1(events, blk_5, blk_4r, miner_account, tx_0); // 13 + 2N
  122. DO_CALLBACK(events, "check_balances_2"); // 14 + 2N
  123. return true;
  124. }
  125. bool gen_ring_signature_2::check_balances_1(cryptonote::core& c, size_t ev_index, const std::vector<test_event_entry>& events)
  126. {
  127. DEFINE_TESTS_ERROR_CONTEXT("gen_ring_signature_2::check_balances_1");
  128. m_bob_account = boost::get<account_base>(events[1]);
  129. m_alice_account = boost::get<account_base>(events[2]);
  130. std::list<block> blocks;
  131. bool r = c.get_blocks(0, 100 + 2 * CRYPTONOTE_MINED_MONEY_UNLOCK_WINDOW, blocks);
  132. CHECK_TEST_CONDITION(r);
  133. std::vector<cryptonote::block> chain;
  134. map_hash2tx_t mtx;
  135. r = find_block_chain(events, chain, mtx, get_block_hash(blocks.back()));
  136. CHECK_TEST_CONDITION(r);
  137. CHECK_EQ(MK_COINS(244), get_balance(m_bob_account, chain, mtx));
  138. CHECK_EQ(0, get_balance(m_alice_account, chain, mtx));
  139. return true;
  140. }
  141. bool gen_ring_signature_2::check_balances_2(cryptonote::core& c, size_t ev_index, const std::vector<test_event_entry>& events)
  142. {
  143. DEFINE_TESTS_ERROR_CONTEXT("gen_ring_signature_2::check_balances_2");
  144. std::list<block> blocks;
  145. bool r = c.get_blocks(0, 100 + 2 * CRYPTONOTE_MINED_MONEY_UNLOCK_WINDOW, blocks);
  146. CHECK_TEST_CONDITION(r);
  147. std::vector<cryptonote::block> chain;
  148. map_hash2tx_t mtx;
  149. r = find_block_chain(events, chain, mtx, get_block_hash(blocks.back()));
  150. CHECK_TEST_CONDITION(r);
  151. CHECK_EQ(0, get_balance(m_bob_account, chain, mtx));
  152. CHECK_EQ(MK_COINS(244) - TESTS_DEFAULT_FEE, get_balance(m_alice_account, chain, mtx));
  153. return true;
  154. }
  155. ////////
  156. // class gen_ring_signature_big;
  157. gen_ring_signature_big::gen_ring_signature_big()
  158. : m_test_size(100)
  159. , m_tx_amount(MK_COINS(29))
  160. {
  161. REGISTER_CALLBACK("check_balances_1", gen_ring_signature_big::check_balances_1);
  162. REGISTER_CALLBACK("check_balances_2", gen_ring_signature_big::check_balances_2);
  163. }
  164. /**
  165. * Check ring signature with m_test_size-1 sources.
  166. * - Create 100 accounts.
  167. * - Create 100 blocks, each block contains transaction from the miner to account[i].
  168. * - Create transaction with ring signature from account[99] to Alice with nmix = 99.
  169. * - Check balances.
  170. */
  171. bool gen_ring_signature_big::generate(std::vector<test_event_entry>& events) const
  172. {
  173. std::vector<account_base> accounts(m_test_size);
  174. std::vector<block> blocks;
  175. blocks.reserve(m_test_size + CRYPTONOTE_MINED_MONEY_UNLOCK_WINDOW);
  176. uint64_t ts_start = 1338224400;
  177. GENERATE_ACCOUNT(miner_account);
  178. MAKE_GENESIS_BLOCK(events, blk_0, miner_account, ts_start);
  179. for (size_t i = 0; i < m_test_size; ++i)
  180. {
  181. MAKE_ACCOUNT(events, an_account);
  182. accounts[i] = an_account;
  183. }
  184. MAKE_ACCOUNT(events, alice_account);
  185. size_t blk_0r_idx = events.size();
  186. REWIND_BLOCKS(events, blk_0r, blk_0, miner_account);
  187. blocks.push_back(blk_0);
  188. for (size_t i = blk_0r_idx; i < events.size(); ++i)
  189. {
  190. blocks.push_back(boost::get<block>(events[i]));
  191. }
  192. for (size_t i = 0; i < m_test_size; ++i)
  193. {
  194. block blk_with_unlocked_out = blocks[blocks.size() - 1 - CRYPTONOTE_MINED_MONEY_UNLOCK_WINDOW];
  195. MAKE_TX_LIST_START(events, txs_blk_i, miner_account, accounts[i], m_tx_amount, blk_with_unlocked_out);
  196. for (size_t j = 0; j <= i; ++j)
  197. {
  198. MAKE_TX_LIST(events, txs_blk_i, miner_account, accounts[i], TESTS_DEFAULT_FEE, blk_with_unlocked_out);
  199. }
  200. MAKE_NEXT_BLOCK_TX_LIST(events, blk_i, blocks.back(), miner_account, txs_blk_i);
  201. blocks.push_back(blk_i);
  202. std::vector<cryptonote::block> chain;
  203. map_hash2tx_t mtx;
  204. bool r = find_block_chain(events, chain, mtx, get_block_hash(blk_i));
  205. CHECK_AND_NO_ASSERT_MES(r, false, "failed to call find_block_chain");
  206. std::cout << i << ": " << get_balance(accounts[i], chain, mtx) << std::endl;
  207. }
  208. DO_CALLBACK(events, "check_balances_1");
  209. MAKE_TX_MIX(events, tx_0, accounts[0], alice_account, m_tx_amount, m_test_size - 1, blocks.back());
  210. MAKE_NEXT_BLOCK_TX1(events, blk_1, blocks.back(), miner_account, tx_0);
  211. DO_CALLBACK(events, "check_balances_2");
  212. return true;
  213. }
  214. bool gen_ring_signature_big::check_balances_1(cryptonote::core& c, size_t ev_index, const std::vector<test_event_entry>& events)
  215. {
  216. DEFINE_TESTS_ERROR_CONTEXT("gen_ring_signature_big::check_balances_1");
  217. m_bob_account = boost::get<account_base>(events[1]);
  218. m_alice_account = boost::get<account_base>(events[1 + m_test_size]);
  219. std::list<block> blocks;
  220. bool r = c.get_blocks(0, 2 * m_test_size + CRYPTONOTE_MINED_MONEY_UNLOCK_WINDOW, blocks);
  221. CHECK_TEST_CONDITION(r);
  222. std::vector<cryptonote::block> chain;
  223. map_hash2tx_t mtx;
  224. r = find_block_chain(events, chain, mtx, get_block_hash(blocks.back()));
  225. CHECK_TEST_CONDITION(r);
  226. CHECK_EQ(m_tx_amount + TESTS_DEFAULT_FEE, get_balance(m_bob_account, chain, mtx));
  227. CHECK_EQ(0, get_balance(m_alice_account, chain, mtx));
  228. for (size_t i = 2; i < 1 + m_test_size; ++i)
  229. {
  230. const account_base& an_account = boost::get<account_base>(events[i]);
  231. uint64_t balance = m_tx_amount + TESTS_DEFAULT_FEE * i;
  232. CHECK_EQ(balance, get_balance(an_account, chain, mtx));
  233. }
  234. return true;
  235. }
  236. bool gen_ring_signature_big::check_balances_2(cryptonote::core& c, size_t ev_index, const std::vector<test_event_entry>& events)
  237. {
  238. DEFINE_TESTS_ERROR_CONTEXT("gen_ring_signature_big::check_balances_2");
  239. std::list<block> blocks;
  240. bool r = c.get_blocks(0, 2 * m_test_size + CRYPTONOTE_MINED_MONEY_UNLOCK_WINDOW, blocks);
  241. CHECK_TEST_CONDITION(r);
  242. std::vector<cryptonote::block> chain;
  243. map_hash2tx_t mtx;
  244. r = find_block_chain(events, chain, mtx, get_block_hash(blocks.back()));
  245. CHECK_TEST_CONDITION(r);
  246. CHECK_EQ(0, get_balance(m_bob_account, chain, mtx));
  247. CHECK_EQ(m_tx_amount, get_balance(m_alice_account, chain, mtx));
  248. for (size_t i = 2; i < 1 + m_test_size; ++i)
  249. {
  250. const account_base& an_account = boost::get<account_base>(events[i]);
  251. uint64_t balance = m_tx_amount + TESTS_DEFAULT_FEE * i;
  252. CHECK_EQ(balance, get_balance(an_account, chain, mtx));
  253. }
  254. std::vector<size_t> tx_outs;
  255. uint64_t transfered;
  256. lookup_acc_outs(m_alice_account.get_keys(), boost::get<transaction>(events[events.size() - 3]), get_tx_pub_key_from_extra(boost::get<transaction>(events[events.size() - 3])), tx_outs, transfered);
  257. CHECK_EQ(m_tx_amount, transfered);
  258. return true;
  259. }