Bitcoin Core  27.1.0
P2P Digital Currency
compress_tests.cpp
Go to the documentation of this file.
1 // Copyright (c) 2012-2021 The Bitcoin Core developers
2 // Distributed under the MIT software license, see the accompanying
3 // file COPYING or http://www.opensource.org/licenses/mit-license.php.
4 
5 #include <compressor.h>
6 #include <script/script.h>
8 
9 #include <stdint.h>
10 
11 #include <boost/test/unit_test.hpp>
12 
13 // amounts 0.00000001 .. 0.00100000
14 #define NUM_MULTIPLES_UNIT 100000
15 
16 // amounts 0.01 .. 100.00
17 #define NUM_MULTIPLES_CENT 10000
18 
19 // amounts 1 .. 10000
20 #define NUM_MULTIPLES_1BTC 10000
21 
22 // amounts 50 .. 21000000
23 #define NUM_MULTIPLES_50BTC 420000
24 
25 BOOST_FIXTURE_TEST_SUITE(compress_tests, BasicTestingSetup)
26 
27 bool static TestEncode(uint64_t in) {
28  return in == DecompressAmount(CompressAmount(in));
29 }
30 
31 bool static TestDecode(uint64_t in) {
32  return in == CompressAmount(DecompressAmount(in));
33 }
34 
35 bool static TestPair(uint64_t dec, uint64_t enc) {
36  return CompressAmount(dec) == enc &&
37  DecompressAmount(enc) == dec;
38 }
39 
40 BOOST_AUTO_TEST_CASE(compress_amounts)
41 {
42  BOOST_CHECK(TestPair( 0, 0x0));
43  BOOST_CHECK(TestPair( 1, 0x1));
44  BOOST_CHECK(TestPair( CENT, 0x7));
45  BOOST_CHECK(TestPair( COIN, 0x9));
46  BOOST_CHECK(TestPair( 50*COIN, 0x32));
47  BOOST_CHECK(TestPair(21000000*COIN, 0x1406f40));
48 
49  for (uint64_t i = 1; i <= NUM_MULTIPLES_UNIT; i++)
51 
52  for (uint64_t i = 1; i <= NUM_MULTIPLES_CENT; i++)
54 
55  for (uint64_t i = 1; i <= NUM_MULTIPLES_1BTC; i++)
57 
58  for (uint64_t i = 1; i <= NUM_MULTIPLES_50BTC; i++)
59  BOOST_CHECK(TestEncode(i * 50 * COIN));
60 
61  for (uint64_t i = 0; i < 100000; i++)
63 }
64 
65 BOOST_AUTO_TEST_CASE(compress_script_to_ckey_id)
66 {
67  // case CKeyID
68  CKey key = GenerateRandomKey();
69  CPubKey pubkey = key.GetPubKey();
70 
71  CScript script = CScript() << OP_DUP << OP_HASH160 << ToByteVector(pubkey.GetID()) << OP_EQUALVERIFY << OP_CHECKSIG;
72  BOOST_CHECK_EQUAL(script.size(), 25U);
73 
75  bool done = CompressScript(script, out);
76  BOOST_CHECK_EQUAL(done, true);
77 
78  // Check compressed script
79  BOOST_CHECK_EQUAL(out.size(), 21U);
80  BOOST_CHECK_EQUAL(out[0], 0x00);
81  BOOST_CHECK_EQUAL(memcmp(out.data() + 1, script.data() + 3, 20), 0); // compare the 20 relevant chars of the CKeyId in the script
82 }
83 
84 BOOST_AUTO_TEST_CASE(compress_script_to_cscript_id)
85 {
86  // case CScriptID
87  CScript script, redeemScript;
88  script << OP_HASH160 << ToByteVector(CScriptID(redeemScript)) << OP_EQUAL;
89  BOOST_CHECK_EQUAL(script.size(), 23U);
90 
92  bool done = CompressScript(script, out);
93  BOOST_CHECK_EQUAL(done, true);
94 
95  // Check compressed script
96  BOOST_CHECK_EQUAL(out.size(), 21U);
97  BOOST_CHECK_EQUAL(out[0], 0x01);
98  BOOST_CHECK_EQUAL(memcmp(out.data() + 1, script.data() + 2, 20), 0); // compare the 20 relevant chars of the CScriptId in the script
99 }
100 
101 BOOST_AUTO_TEST_CASE(compress_script_to_compressed_pubkey_id)
102 {
103  CKey key = GenerateRandomKey(); // case compressed PubKeyID
104 
105  CScript script = CScript() << ToByteVector(key.GetPubKey()) << OP_CHECKSIG; // COMPRESSED_PUBLIC_KEY_SIZE (33)
106  BOOST_CHECK_EQUAL(script.size(), 35U);
107 
109  bool done = CompressScript(script, out);
110  BOOST_CHECK_EQUAL(done, true);
111 
112  // Check compressed script
113  BOOST_CHECK_EQUAL(out.size(), 33U);
114  BOOST_CHECK_EQUAL(memcmp(out.data(), script.data() + 1, 1), 0);
115  BOOST_CHECK_EQUAL(memcmp(out.data() + 1, script.data() + 2, 32), 0); // compare the 32 chars of the compressed CPubKey
116 }
117 
118 BOOST_AUTO_TEST_CASE(compress_script_to_uncompressed_pubkey_id)
119 {
120  CKey key = GenerateRandomKey(/*compressed=*/false); // case uncompressed PubKeyID
121  CScript script = CScript() << ToByteVector(key.GetPubKey()) << OP_CHECKSIG; // PUBLIC_KEY_SIZE (65)
122  BOOST_CHECK_EQUAL(script.size(), 67U); // 1 char code + 65 char pubkey + OP_CHECKSIG
123 
125  bool done = CompressScript(script, out);
126  BOOST_CHECK_EQUAL(done, true);
127 
128  // Check compressed script
129  BOOST_CHECK_EQUAL(out.size(), 33U);
130  BOOST_CHECK_EQUAL(memcmp(out.data() + 1, script.data() + 2, 32), 0); // first 32 chars of CPubKey are copied into out[1:]
131  BOOST_CHECK_EQUAL(out[0], 0x04 | (script[65] & 0x01)); // least significant bit (lsb) of last char of pubkey is mapped into out[0]
132 }
133 
uint64_t CompressAmount(uint64_t n)
Compress amount.
Definition: compressor.cpp:149
BOOST_AUTO_TEST_CASE(compress_amounts)
static bool TestEncode(uint64_t in)
Definition: script.h:124
CPubKey GetPubKey() const
Compute the public key from a private key.
Definition: key.cpp:188
CKeyID GetID() const
Get the KeyID of this public key (hash of its serialization)
Definition: pubkey.h:164
value_type * data()
Definition: prevector.h:532
Basic testing setup.
Definition: setup_common.h:49
CKey GenerateRandomKey(bool compressed) noexcept
Definition: key.cpp:372
#define NUM_MULTIPLES_CENT
#define NUM_MULTIPLES_UNIT
static bool TestPair(uint64_t dec, uint64_t enc)
#define NUM_MULTIPLES_1BTC
bool CompressScript(const CScript &script, CompressedScript &out)
Definition: compressor.cpp:55
BOOST_AUTO_TEST_SUITE_END()
An encapsulated public key.
Definition: pubkey.h:33
#define NUM_MULTIPLES_50BTC
Implements a drop-in replacement for std::vector<T> which stores up to N elements directly (without h...
Definition: prevector.h:37
std::vector< unsigned char > ToByteVector(const T &in)
Definition: script.h:66
#define BOOST_CHECK_EQUAL(v1, v2)
Definition: object.cpp:18
Serialized script, used inside transaction inputs and outputs.
Definition: script.h:413
A reference to a CScript: the Hash160 of its serialization.
Definition: script.h:582
size_type size() const
Definition: prevector.h:296
static constexpr CAmount CENT
Definition: setup_common.h:44
An encapsulated private key.
Definition: key.h:32
uint64_t DecompressAmount(uint64_t x)
Definition: compressor.cpp:168
static bool TestDecode(uint64_t in)
#define BOOST_CHECK(expr)
Definition: object.cpp:17
static constexpr CAmount COIN
The amount of satoshis in one BTC.
Definition: amount.h:15