LCOV - code coverage report
Current view: top level - src/test - bloom_tests.cpp (source / functions) Hit Total Coverage
Test: total_coverage.info Lines: 389 389 100.0 %
Date: 2026-06-25 07:23:43 Functions: 102 102 100.0 %

          Line data    Source code
       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 <bls/bls.h>
       6             : #include <clientversion.h>
       7             : #include <common/bloom.h>
       8             : #include <key.h>
       9             : #include <key_io.h>
      10             : #include <merkleblock.h>
      11             : #include <primitives/block.h>
      12             : #include <serialize.h>
      13             : #include <streams.h>
      14             : #include <test/util/random.h>
      15             : #include <test/util/setup_common.h>
      16             : #include <uint256.h>
      17             : #include <util/strencodings.h>
      18             : 
      19             : #include <vector>
      20             : 
      21             : #include <boost/test/unit_test.hpp>
      22             : 
      23         146 : BOOST_FIXTURE_TEST_SUITE(bloom_tests, BasicTestingSetup)
      24             : 
      25         149 : BOOST_AUTO_TEST_CASE(bloom_create_insert_serialize)
      26             : {
      27           1 :     CBloomFilter filter(3, 0.01, 0, BLOOM_UPDATE_ALL);
      28             : 
      29           1 :     BOOST_CHECK_MESSAGE( !filter.contains(ParseHex("99108ad8ed9bb6274d3980bab5a85c048f0950c8")), "Bloom filter should be empty!");
      30           1 :     filter.insert(ParseHex("99108ad8ed9bb6274d3980bab5a85c048f0950c8"));
      31           1 :     BOOST_CHECK_MESSAGE( filter.contains(ParseHex("99108ad8ed9bb6274d3980bab5a85c048f0950c8")), "Bloom filter doesn't contain just-inserted object!");
      32             :     // One bit different in first byte
      33           1 :     BOOST_CHECK_MESSAGE(!filter.contains(ParseHex("19108ad8ed9bb6274d3980bab5a85c048f0950c8")), "Bloom filter contains something it shouldn't!");
      34             : 
      35           1 :     filter.insert(ParseHex("b5a2c786d9ef4658287ced5914b37a1b4aa32eee"));
      36           1 :     BOOST_CHECK_MESSAGE(filter.contains(ParseHex("b5a2c786d9ef4658287ced5914b37a1b4aa32eee")), "Bloom filter doesn't contain just-inserted object (2)!");
      37             : 
      38           1 :     filter.insert(ParseHex("b9300670b4c5366e95b2699e8b18bc75e5f729c5"));
      39           1 :     BOOST_CHECK_MESSAGE(filter.contains(ParseHex("b9300670b4c5366e95b2699e8b18bc75e5f729c5")), "Bloom filter doesn't contain just-inserted object (3)!");
      40             : 
      41           1 :     CDataStream stream(SER_NETWORK, PROTOCOL_VERSION);
      42           1 :     stream << filter;
      43             : 
      44           1 :     std::vector<uint8_t> expected = ParseHex("03614e9b050000000000000001");
      45           1 :     auto result{MakeUCharSpan(stream)};
      46             : 
      47           1 :     BOOST_CHECK_EQUAL_COLLECTIONS(result.begin(), result.end(), expected.begin(), expected.end());
      48             : 
      49           1 :     BOOST_CHECK_MESSAGE( filter.contains(ParseHex("99108ad8ed9bb6274d3980bab5a85c048f0950c8")), "Bloom filter doesn't contain just-inserted object!");
      50           1 : }
      51             : 
      52         149 : BOOST_AUTO_TEST_CASE(bloom_create_insert_serialize_with_tweak)
      53             : {
      54             :     // Same test as bloom_create_insert_serialize, but we add a nTweak of 100
      55           1 :     CBloomFilter filter(3, 0.01, 2147483649UL, BLOOM_UPDATE_ALL);
      56             : 
      57           1 :     filter.insert(ParseHex("99108ad8ed9bb6274d3980bab5a85c048f0950c8"));
      58           1 :     BOOST_CHECK_MESSAGE( filter.contains(ParseHex("99108ad8ed9bb6274d3980bab5a85c048f0950c8")), "Bloom filter doesn't contain just-inserted object!");
      59             :     // One bit different in first byte
      60           1 :     BOOST_CHECK_MESSAGE(!filter.contains(ParseHex("19108ad8ed9bb6274d3980bab5a85c048f0950c8")), "Bloom filter contains something it shouldn't!");
      61             : 
      62           1 :     filter.insert(ParseHex("b5a2c786d9ef4658287ced5914b37a1b4aa32eee"));
      63           1 :     BOOST_CHECK_MESSAGE(filter.contains(ParseHex("b5a2c786d9ef4658287ced5914b37a1b4aa32eee")), "Bloom filter doesn't contain just-inserted object (2)!");
      64             : 
      65           1 :     filter.insert(ParseHex("b9300670b4c5366e95b2699e8b18bc75e5f729c5"));
      66           1 :     BOOST_CHECK_MESSAGE(filter.contains(ParseHex("b9300670b4c5366e95b2699e8b18bc75e5f729c5")), "Bloom filter doesn't contain just-inserted object (3)!");
      67             : 
      68           1 :     CDataStream stream(SER_NETWORK, PROTOCOL_VERSION);
      69           1 :     stream << filter;
      70             : 
      71           1 :     std::vector<uint8_t> expected = ParseHex("03ce4299050000000100008001");
      72           1 :     auto result{MakeUCharSpan(stream)};
      73             : 
      74           1 :     BOOST_CHECK_EQUAL_COLLECTIONS(result.begin(), result.end(), expected.begin(), expected.end());
      75           1 : }
      76             : 
      77         149 : BOOST_AUTO_TEST_CASE(bloom_create_insert_key)
      78             : {
      79           1 :     std::string strSecret = std::string("7sQb6QHALg4XyHsJHsSNXnEHGhZfzTTUPJXJqaqK7CavQkiL9Ms");
      80           1 :     CKey key = DecodeSecret(strSecret);
      81           1 :     CPubKey pubkey = key.GetPubKey();
      82           1 :     std::vector<unsigned char> vchPubKey(pubkey.begin(), pubkey.end());
      83             : 
      84           1 :     CBloomFilter filter(2, 0.001, 0, BLOOM_UPDATE_ALL);
      85           1 :     filter.insert(vchPubKey);
      86           1 :     uint160 hash = pubkey.GetID();
      87           1 :     filter.insert(hash);
      88             : 
      89           1 :     CDataStream stream(SER_NETWORK, PROTOCOL_VERSION);
      90           1 :     stream << filter;
      91             : 
      92           1 :     std::vector<unsigned char> expected = ParseHex("038fc16b080000000000000001");
      93           1 :     auto result{MakeUCharSpan(stream)};
      94             : 
      95           1 :     BOOST_CHECK_EQUAL_COLLECTIONS(result.begin(), result.end(), expected.begin(), expected.end());
      96           1 : }
      97             : 
      98         149 : BOOST_AUTO_TEST_CASE(bloom_match)
      99             : {
     100             :     // Random real transaction (b4749f017444b051c44dfd2720e88f314ff94f3dd6d56d40ef65854fcd7fff6b)
     101           1 :     CDataStream stream(ParseHex("01000000010b26e9b7735eb6aabdf358bab62f9816a21ba9ebdb719d5299e88607d722c190000000008b4830450220070aca44506c5cef3a16ed519d7c3c39f8aab192c4e1c90d065f37b8a4af6141022100a8e160b856c2d43d27d8fba71e5aef6405b8643ac4cb7cb3c462aced7f14711a0141046d11fee51b0e60666d5049a9101a72741df480b96ee26488a4d3466b95c9a40ac5eeef87e10a5cd336c19a84565f80fa6c547957b7700ff4dfbdefe76036c339ffffffff021bff3d11000000001976a91404943fdd508053c75000106d3bc6e2754dbcff1988ac2f15de00000000001976a914a266436d2965547608b9e15d9032a7b9d64fa43188ac00000000"), SER_DISK, CLIENT_VERSION);
     102           1 :     CTransaction tx(deserialize, stream);
     103             : 
     104             :     // and one which spends it (e2769b09e784f32f62ef849763d4f45b98e07ba658647343b915ff832b110436)
     105           1 :     unsigned char ch[] = {0x01, 0x00, 0x00, 0x00, 0x01, 0x6b, 0xff, 0x7f, 0xcd, 0x4f, 0x85, 0x65, 0xef, 0x40, 0x6d, 0xd5, 0xd6, 0x3d, 0x4f, 0xf9, 0x4f, 0x31, 0x8f, 0xe8, 0x20, 0x27, 0xfd, 0x4d, 0xc4, 0x51, 0xb0, 0x44, 0x74, 0x01, 0x9f, 0x74, 0xb4, 0x00, 0x00, 0x00, 0x00, 0x8c, 0x49, 0x30, 0x46, 0x02, 0x21, 0x00, 0xda, 0x0d, 0xc6, 0xae, 0xce, 0xfe, 0x1e, 0x06, 0xef, 0xdf, 0x05, 0x77, 0x37, 0x57, 0xde, 0xb1, 0x68, 0x82, 0x09, 0x30, 0xe3, 0xb0, 0xd0, 0x3f, 0x46, 0xf5, 0xfc, 0xf1, 0x50, 0xbf, 0x99, 0x0c, 0x02, 0x21, 0x00, 0xd2, 0x5b, 0x5c, 0x87, 0x04, 0x00, 0x76, 0xe4, 0xf2, 0x53, 0xf8, 0x26, 0x2e, 0x76, 0x3e, 0x2d, 0xd5, 0x1e, 0x7f, 0xf0, 0xbe, 0x15, 0x77, 0x27, 0xc4, 0xbc, 0x42, 0x80, 0x7f, 0x17, 0xbd, 0x39, 0x01, 0x41, 0x04, 0xe6, 0xc2, 0x6e, 0xf6, 0x7d, 0xc6, 0x10, 0xd2, 0xcd, 0x19, 0x24, 0x84, 0x78, 0x9a, 0x6c, 0xf9, 0xae, 0xa9, 0x93, 0x0b, 0x94, 0x4b, 0x7e, 0x2d, 0xb5, 0x34, 0x2b, 0x9d, 0x9e, 0x5b, 0x9f, 0xf7, 0x9a, 0xff, 0x9a, 0x2e, 0xe1, 0x97, 0x8d, 0xd7, 0xfd, 0x01, 0xdf, 0xc5, 0x22, 0xee, 0x02, 0x28, 0x3d, 0x3b, 0x06, 0xa9, 0xd0, 0x3a, 0xcf, 0x80, 0x96, 0x96, 0x8d, 0x7d, 0xbb, 0x0f, 0x91, 0x78, 0xff, 0xff, 0xff, 0xff, 0x02, 0x8b, 0xa7, 0x94, 0x0e, 0x00, 0x00, 0x00, 0x00, 0x19, 0x76, 0xa9, 0x14, 0xba, 0xde, 0xec, 0xfd, 0xef, 0x05, 0x07, 0x24, 0x7f, 0xc8, 0xf7, 0x42, 0x41, 0xd7, 0x3b, 0xc0, 0x39, 0x97, 0x2d, 0x7b, 0x88, 0xac, 0x40, 0x94, 0xa8, 0x02, 0x00, 0x00, 0x00, 0x00, 0x19, 0x76, 0xa9, 0x14, 0xc1, 0x09, 0x32, 0x48, 0x3f, 0xec, 0x93, 0xed, 0x51, 0xf5, 0xfe, 0x95, 0xe7, 0x25, 0x59, 0xf2, 0xcc, 0x70, 0x43, 0xf9, 0x88, 0xac, 0x00, 0x00, 0x00, 0x00, 0x00};
     106           1 :     std::vector<unsigned char> vch(ch, ch + sizeof(ch) -1);
     107           1 :     CDataStream spendStream(vch, SER_DISK, CLIENT_VERSION);
     108           1 :     CTransaction spendingTx(deserialize, spendStream);
     109             : 
     110           1 :     CBloomFilter filter(10, 0.000001, 0, BLOOM_UPDATE_ALL);
     111           1 :     filter.insert(uint256S("0xb4749f017444b051c44dfd2720e88f314ff94f3dd6d56d40ef65854fcd7fff6b"));
     112           1 :     BOOST_CHECK_MESSAGE(filter.IsRelevantAndUpdate(tx), "Simple Bloom filter didn't match tx hash");
     113             : 
     114           1 :     filter = CBloomFilter(10, 0.000001, 0, BLOOM_UPDATE_ALL);
     115             :     // byte-reversed tx hash
     116           1 :     filter.insert(ParseHex("6bff7fcd4f8565ef406dd5d63d4ff94f318fe82027fd4dc451b04474019f74b4"));
     117           1 :     BOOST_CHECK_MESSAGE(filter.IsRelevantAndUpdate(tx), "Simple Bloom filter didn't match manually serialized tx hash");
     118             : 
     119           1 :     filter = CBloomFilter(10, 0.000001, 0, BLOOM_UPDATE_ALL);
     120           1 :     filter.insert(ParseHex("30450220070aca44506c5cef3a16ed519d7c3c39f8aab192c4e1c90d065f37b8a4af6141022100a8e160b856c2d43d27d8fba71e5aef6405b8643ac4cb7cb3c462aced7f14711a01"));
     121           1 :     BOOST_CHECK_MESSAGE(filter.IsRelevantAndUpdate(tx), "Simple Bloom filter didn't match input signature");
     122             : 
     123           1 :     filter = CBloomFilter(10, 0.000001, 0, BLOOM_UPDATE_ALL);
     124           1 :     filter.insert(ParseHex("046d11fee51b0e60666d5049a9101a72741df480b96ee26488a4d3466b95c9a40ac5eeef87e10a5cd336c19a84565f80fa6c547957b7700ff4dfbdefe76036c339"));
     125           1 :     BOOST_CHECK_MESSAGE(filter.IsRelevantAndUpdate(tx), "Simple Bloom filter didn't match input pub key");
     126             : 
     127           1 :     filter = CBloomFilter(10, 0.000001, 0, BLOOM_UPDATE_ALL);
     128           1 :     filter.insert(ParseHex("04943fdd508053c75000106d3bc6e2754dbcff19"));
     129           1 :     BOOST_CHECK_MESSAGE(filter.IsRelevantAndUpdate(tx), "Simple Bloom filter didn't match output address");
     130           1 :     BOOST_CHECK_MESSAGE(filter.IsRelevantAndUpdate(spendingTx), "Simple Bloom filter didn't add output");
     131             : 
     132           1 :     filter = CBloomFilter(10, 0.000001, 0, BLOOM_UPDATE_ALL);
     133           1 :     filter.insert(ParseHex("a266436d2965547608b9e15d9032a7b9d64fa431"));
     134           1 :     BOOST_CHECK_MESSAGE(filter.IsRelevantAndUpdate(tx), "Simple Bloom filter didn't match output address");
     135             : 
     136           1 :     filter = CBloomFilter(10, 0.000001, 0, BLOOM_UPDATE_ALL);
     137           1 :     filter.insert(COutPoint(uint256S("0x90c122d70786e899529d71dbeba91ba216982fb6ba58f3bdaab65e73b7e9260b"), 0));
     138           1 :     BOOST_CHECK_MESSAGE(filter.IsRelevantAndUpdate(tx), "Simple Bloom filter didn't match COutPoint");
     139             : 
     140           1 :     filter = CBloomFilter(10, 0.000001, 0, BLOOM_UPDATE_ALL);
     141           1 :     COutPoint prevOutPoint(uint256S("0x90c122d70786e899529d71dbeba91ba216982fb6ba58f3bdaab65e73b7e9260b"), 0);
     142             :     {
     143           1 :         std::vector<unsigned char> data(32 + sizeof(unsigned int));
     144           1 :         memcpy(data.data(), prevOutPoint.hash.begin(), 32);
     145           1 :         memcpy(data.data()+32, &prevOutPoint.n, sizeof(unsigned int));
     146           1 :         filter.insert(data);
     147           1 :     }
     148           1 :     BOOST_CHECK_MESSAGE(filter.IsRelevantAndUpdate(tx), "Simple Bloom filter didn't match manually serialized COutPoint");
     149             : 
     150           1 :     filter = CBloomFilter(10, 0.000001, 0, BLOOM_UPDATE_ALL);
     151           1 :     filter.insert(uint256S("00000009e784f32f62ef849763d4f45b98e07ba658647343b915ff832b110436"));
     152           1 :     BOOST_CHECK_MESSAGE(!filter.IsRelevantAndUpdate(tx), "Simple Bloom filter matched random tx hash");
     153             : 
     154           1 :     filter = CBloomFilter(10, 0.000001, 0, BLOOM_UPDATE_ALL);
     155           1 :     filter.insert(ParseHex("0000006d2965547608b9e15d9032a7b9d64fa431"));
     156           1 :     BOOST_CHECK_MESSAGE(!filter.IsRelevantAndUpdate(tx), "Simple Bloom filter matched random address");
     157             : 
     158           1 :     filter = CBloomFilter(10, 0.000001, 0, BLOOM_UPDATE_ALL);
     159           1 :     filter.insert(COutPoint(uint256S("0x90c122d70786e899529d71dbeba91ba216982fb6ba58f3bdaab65e73b7e9260b"), 1));
     160           1 :     BOOST_CHECK_MESSAGE(!filter.IsRelevantAndUpdate(tx), "Simple Bloom filter matched COutPoint for an output we didn't care about");
     161             : 
     162           1 :     filter = CBloomFilter(10, 0.000001, 0, BLOOM_UPDATE_ALL);
     163           1 :     filter.insert(COutPoint(uint256S("0x000000d70786e899529d71dbeba91ba216982fb6ba58f3bdaab65e73b7e9260b"), 0));
     164           1 :     BOOST_CHECK_MESSAGE(!filter.IsRelevantAndUpdate(tx), "Simple Bloom filter matched COutPoint for an output we didn't care about");
     165           1 : }
     166             : 
     167         149 : BOOST_AUTO_TEST_CASE(dip2_bloom_match)
     168             : {
     169             :     //TODO: Provide raw data for basic scheme as well
     170           1 :     bls::bls_legacy_scheme.store(true);
     171             :     // ProRegTx from testnet (txid: 39a1339d9bf26de701345beecc5de75a690bc9533741a3dbe90f2fd88b8ed461)
     172           1 :     CDataStream stream(ParseHex("0300010001c02142f16969d106f4cb144e17975bf9261b8e331354879c6d006e71494b3460000000006b483045022100d6eb9e25fc3215e6a4b5180a1957319d6658103566f0274452bd50d0eddedff202200824a00f582a29c9e3694d34e5525275cabd08c488a0c55b4350c6bfa4fb2cf8012102a8d6433a8f799a13d3495f4aa5b99bd288adca5f59e0c9609f15a221220241c5feffffff0121c89a3b000000001976a914e54445646929fac8b7d6c71715913af44324978488ac00000000fd12010100000000009b9054ff7839b940277b8eb8211570b2f16850ef729ee635e24d722fbc4a46230100000000000000000000000000ffffc6c74af14e1f891cbc8a94fa7fea64ca9994870dca0f75bbd0750efda51589f86e30cc2305e7388c01ce0309c19a182cf37bced97c7da72236f660c0a395e765e6e06962ecff5a69d7de359c348a574176c210c37a25d4ffd917866fb0a300001976a914e54445646929fac8b7d6c71715913af44324978488ac26d5a99c01521d1fca6299de576bae82f901fb56e4b797a945e876ac69068f36411f9cdb72a01b273a53bd916d8c90dd584bd3a1c01dd84fec84eb046f66a4e3b30d39ea3215293a035166b5072349ebc08efd30c2c73ebd023d920f1db0aef91a4e"), SER_DISK, CLIENT_VERSION);
     173           1 :     CTransaction proregtx(deserialize, stream);
     174             :     // ProUpServTx from testnet (txid: 0c9627a054784f207db34c311b06e8bb797c4a120a236dbb02b51268dd186d39)
     175           1 :     stream = CDataStream(ParseHex("030002000161d48e8bd82f0fe9dba3413753c90b695ae75dccee5b3401e76df29b9d33a139000000006a473044022036dfd41268e3a387b909d9d1cf61e4d5e97b6b37d5951e82c4ca4fed419b4b6a02203777d818319897a11cf4cd8b4c6495c5c3a5f96fcad6d3a0596431b1146b53c6012102a8d6433a8f799a13d3495f4aa5b99bd288adca5f59e0c9609f15a221220241c5feffffff01a1c69a3b000000001976a914e54445646929fac8b7d6c71715913af44324978488ac00000000b5010061d48e8bd82f0fe9dba3413753c90b695ae75dccee5b3401e76df29b9d33a13900000000000000000000ffffc6c74af14e1f000ccf39b295938f4021fc16b23e4eedbd64a6c9a5aaae6bc9ca085db4bc29ff911612dc4544fc470415cdd4be5659fee6aa7d84c3572c712f5904b2e5d38cb217762717d88e4e1a956a9e82ce132e2b9b144e26cb3ff1e53675ede2769d99f46796ea0b8cbf33b78c15fe9a437dc4d1131ce2af8fd2ed5b306b326f9fcffcd416"), SER_DISK, CLIENT_VERSION);
     176           1 :     CTransaction proupservtx(deserialize, stream);
     177             :     // ProUpRegTx from testnet (txid: 3037bf4e45d1a9d57891ea8baa72b38ad6323bd28945b11325bdd33ebea049b3)
     178           1 :     stream = CDataStream(ParseHex("0300030001396d18dd6812b502bb6d230a124a7c79bbe8061b314cb37d204f7854a027960c000000006a47304402203aefa0e1a08469050edc1e5da6e24f783162bebc15cc006f717097e56e07fa0b02207c7f47d113355b095c7edd2408d0554aa6b87187bf95652c516fc7747720fb19012102a8d6433a8f799a13d3495f4aa5b99bd288adca5f59e0c9609f15a221220241c5feffffff01f2c49a3b000000001976a914e54445646929fac8b7d6c71715913af44324978488ac00000000e4010061d48e8bd82f0fe9dba3413753c90b695ae75dccee5b3401e76df29b9d33a13900000efda51589f86e30cc2305e7388c01ce0309c19a182cf37bced97c7da72236f660c0a395e765e6e06962ecff5a69d7de359c348a574176c210c37a25d4ffd917866fb0a31976a914e54445646929fac8b7d6c71715913af44324978488aca8141db635af714f32c7415b298937e171c6bb8a3c8627852ae2b69917ece86a411fa51861025bb5ff4ba4ad3be6090c4ba76b1671d70799ed6882c57bcfeaf27cbd558068e9b5fc1553abce8822fcee63e8f6fbb06ad5b753d47e794bfacadbde3f"), SER_DISK, CLIENT_VERSION);
     179           1 :     CTransaction proupregtx(deserialize, stream);
     180             :     // ProUpRevTx from testnet (txid: 70c41a5fea07a80c1297771c43eb58b8f52222b971f61f5c7c2030ec0baf7f9a)
     181           1 :     stream = CDataStream(ParseHex("0300040001b349a0be3ed3bd2513b14589d23b32d68ab372aa8bea9178d5a9d1454ebf3730000000006a4730440220428d87daf3b0ddca9656d59592000f9930a88162c0b017ca460c69c33b76bda202200276a521d94acc57e010262aa1d7d016cebbe7a9806d26dbebb69102ce9cf7b2012102a8d6433a8f799a13d3495f4aa5b99bd288adca5f59e0c9609f15a221220241c5feffffff0183c39a3b000000001976a914e54445646929fac8b7d6c71715913af44324978488ac00000000a4010061d48e8bd82f0fe9dba3413753c90b695ae75dccee5b3401e76df29b9d33a1390000c024ffd34d15686f7a376ecdf45800ddcebf4eb966fd3123cf9450515a38881a0de852c97297bbd9cedb79280f3f1e31b092904a8870274a5a0e52de2ae1dc0bad74851dc0a1e4fe4c274535a2422d3e1510a7be2fbecaae73128e8eb2f336382fb376d2f82273a72960980972b02e3ecf895c00187e0cbe735dc44cb3d97711"), SER_DISK, CLIENT_VERSION);
     182           1 :     CTransaction prouprevtx(deserialize, stream);
     183             : 
     184             :     // check collateral outpoint match in ProRegTx
     185           1 :     CBloomFilter filter(10, 0.000001, 0, BLOOM_UPDATE_ALL);
     186           1 :     filter.insert(COutPoint(uint256S("0x23464abc2f724de235e69e72ef5068f1b2701521b88e7b2740b93978ff54909b"), 1));
     187           1 :     BOOST_CHECK_MESSAGE(filter.IsRelevantAndUpdate(proregtx), "Bloom filter didn't match collateral outpoint in ProRegTx");
     188             :     // check owner keyid match in ProRegTx
     189           1 :     filter = CBloomFilter(10, 0.000001, 0, BLOOM_UPDATE_ALL);
     190           1 :     filter.insert(ParseHex("891cbc8a94fa7fea64ca9994870dca0f75bbd075"));
     191           1 :     BOOST_CHECK_MESSAGE(filter.IsRelevantAndUpdate(proregtx), "Bloom filter didn't match Owner keyid in ProRegTx");
     192             :     // check voting keyid match in ProRegTx
     193           1 :     filter = CBloomFilter(10, 0.000001, 0, BLOOM_UPDATE_ALL);
     194           1 :     filter.insert(ParseHex("359c348a574176c210c37a25d4ffd917866fb0a3"));
     195           1 :     BOOST_CHECK_MESSAGE(filter.IsRelevantAndUpdate(proregtx), "Bloom filter didn't match Voting keyid in ProRegTx");
     196             :     // check scriptPayout match in ProRegTx
     197           1 :     filter = CBloomFilter(10, 0.000001, 0, BLOOM_UPDATE_ALL);
     198           1 :     filter.insert(ParseHex("e54445646929fac8b7d6c71715913af443249784"));
     199           1 :     BOOST_CHECK_MESSAGE(filter.IsRelevantAndUpdate(proregtx), "Bloom filter didn't match scriptPayout in ProRegTx");
     200             : 
     201             :     // check proTxHash match in ProUpServTx
     202           1 :     filter = CBloomFilter(10, 0.000001, 0, BLOOM_UPDATE_ALL);
     203           1 :     filter.insert(uint256S("0x39a1339d9bf26de701345beecc5de75a690bc9533741a3dbe90f2fd88b8ed461"));
     204           1 :     BOOST_CHECK_MESSAGE(filter.IsRelevantAndUpdate(proupservtx), "Bloom filter didn't match proTxHash in ProUpServTx");
     205             :     // check scriptOperatorPayout match in ProUpServTx
     206           1 :     filter = CBloomFilter(10, 0.000001, 0, BLOOM_UPDATE_ALL);
     207           1 :     filter.insert(ParseHex("e54445646929fac8b7d6c71715913af443249784"));
     208           1 :     BOOST_CHECK_MESSAGE(filter.IsRelevantAndUpdate(proupservtx), "Bloom filter didn't match scriptOperatorPayout in ProUpServTx");
     209             : 
     210             :     // check proTxHash match in ProUpRegTx
     211           1 :     filter = CBloomFilter(10, 0.000001, 0, BLOOM_UPDATE_ALL);
     212           1 :     filter.insert(uint256S("0x39a1339d9bf26de701345beecc5de75a690bc9533741a3dbe90f2fd88b8ed461"));
     213           1 :     BOOST_CHECK_MESSAGE(filter.IsRelevantAndUpdate(proupregtx), "Bloom filter didn't match proTxHash in ProUpRegTx");
     214             :     // check voting keyid match in ProUpRegTx
     215           1 :     filter = CBloomFilter(10, 0.000001, 0, BLOOM_UPDATE_ALL);
     216           1 :     filter.insert(ParseHex("359c348a574176c210c37a25d4ffd917866fb0a3"));
     217           1 :     BOOST_CHECK_MESSAGE(filter.IsRelevantAndUpdate(proupregtx), "Bloom filter didn't match Voting keyid in ProUpRegTx");
     218             :     // check scriptPayout match in ProUpRegTx
     219           1 :     filter = CBloomFilter(10, 0.000001, 0, BLOOM_UPDATE_ALL);
     220           1 :     filter.insert(ParseHex("e54445646929fac8b7d6c71715913af443249784"));
     221           1 :     BOOST_CHECK_MESSAGE(filter.IsRelevantAndUpdate(proupregtx), "Bloom filter didn't match scriptPayout in ProUpRegTx");
     222             : 
     223             :     // check proTxHash match in ProUpRevTx
     224           1 :     filter = CBloomFilter(10, 0.000001, 0, BLOOM_UPDATE_ALL);
     225           1 :     filter.insert(uint256S("0x39a1339d9bf26de701345beecc5de75a690bc9533741a3dbe90f2fd88b8ed461"));
     226           1 :     BOOST_CHECK_MESSAGE(filter.IsRelevantAndUpdate(prouprevtx), "Bloom filter didn't match proTxHash in ProUpRevTx");
     227             : 
     228             :     // check filter is not matching if it doesn't contain relative data
     229           1 :     filter = CBloomFilter(10, 0.000001, 0, BLOOM_UPDATE_ALL);
     230             :     // extend real data with additional byte
     231           1 :     filter.insert(uint256S("0x39a1339d9bf26de701345beecc5de75a690bc9533741a3dbe90f2fd88b8ed46100"));
     232           1 :     filter.insert(ParseHex("e54445646929fac8b7d6c71715913af44324978400"));
     233           1 :     filter.insert(ParseHex("359c348a574176c210c37a25d4ffd917866fb0a300"));
     234           1 :     filter.insert(uint256S("0x39a1339d9bf26de701345beecc5de75a690bc9533741a3dbe90f2fd88b8ed46100"));
     235           1 :     BOOST_CHECK_MESSAGE(!filter.IsRelevantAndUpdate(prouprevtx), "Bloom filter match unrelated data");
     236           1 :     BOOST_CHECK_MESSAGE(!filter.IsRelevantAndUpdate(proupregtx), "Bloom filter match unrelated data");
     237           1 : }
     238             : 
     239         149 : BOOST_AUTO_TEST_CASE(dip2_bloom_update)
     240             : {
     241             :     //TODO: Provide raw data for basic scheme as well
     242           1 :     bls::bls_legacy_scheme.store(true);
     243             :     // ProRegTx from testnet (txid: 39a1339d9bf26de701345beecc5de75a690bc9533741a3dbe90f2fd88b8ed461)
     244           1 :     CDataStream stream(ParseHex("0300010001c02142f16969d106f4cb144e17975bf9261b8e331354879c6d006e71494b3460000000006b483045022100d6eb9e25fc3215e6a4b5180a1957319d6658103566f0274452bd50d0eddedff202200824a00f582a29c9e3694d34e5525275cabd08c488a0c55b4350c6bfa4fb2cf8012102a8d6433a8f799a13d3495f4aa5b99bd288adca5f59e0c9609f15a221220241c5feffffff0121c89a3b000000001976a914e54445646929fac8b7d6c71715913af44324978488ac00000000fd12010100000000009b9054ff7839b940277b8eb8211570b2f16850ef729ee635e24d722fbc4a46230100000000000000000000000000ffffc6c74af14e1f891cbc8a94fa7fea64ca9994870dca0f75bbd0750efda51589f86e30cc2305e7388c01ce0309c19a182cf37bced97c7da72236f660c0a395e765e6e06962ecff5a69d7de359c348a574176c210c37a25d4ffd917866fb0a300001976a914e54445646929fac8b7d6c71715913af44324978488ac26d5a99c01521d1fca6299de576bae82f901fb56e4b797a945e876ac69068f36411f9cdb72a01b273a53bd916d8c90dd584bd3a1c01dd84fec84eb046f66a4e3b30d39ea3215293a035166b5072349ebc08efd30c2c73ebd023d920f1db0aef91a4e"), SER_DISK, CLIENT_VERSION);
     245           1 :     CTransaction proregtx(deserialize, stream);
     246             :     // ProUpServTx from testnet (txid: 0c9627a054784f207db34c311b06e8bb797c4a120a236dbb02b51268dd186d39)
     247           1 :     stream = CDataStream(ParseHex("030002000161d48e8bd82f0fe9dba3413753c90b695ae75dccee5b3401e76df29b9d33a139000000006a473044022036dfd41268e3a387b909d9d1cf61e4d5e97b6b37d5951e82c4ca4fed419b4b6a02203777d818319897a11cf4cd8b4c6495c5c3a5f96fcad6d3a0596431b1146b53c6012102a8d6433a8f799a13d3495f4aa5b99bd288adca5f59e0c9609f15a221220241c5feffffff01a1c69a3b000000001976a914e54445646929fac8b7d6c71715913af44324978488ac00000000b5010061d48e8bd82f0fe9dba3413753c90b695ae75dccee5b3401e76df29b9d33a13900000000000000000000ffffc6c74af14e1f000ccf39b295938f4021fc16b23e4eedbd64a6c9a5aaae6bc9ca085db4bc29ff911612dc4544fc470415cdd4be5659fee6aa7d84c3572c712f5904b2e5d38cb217762717d88e4e1a956a9e82ce132e2b9b144e26cb3ff1e53675ede2769d99f46796ea0b8cbf33b78c15fe9a437dc4d1131ce2af8fd2ed5b306b326f9fcffcd416"), SER_DISK, CLIENT_VERSION);
     248           1 :     CTransaction proupservtx(deserialize, stream);
     249             :     // ProUpRegTx from testnet (txid: 3037bf4e45d1a9d57891ea8baa72b38ad6323bd28945b11325bdd33ebea049b3)
     250           1 :     stream = CDataStream(ParseHex("0300030001396d18dd6812b502bb6d230a124a7c79bbe8061b314cb37d204f7854a027960c000000006a47304402203aefa0e1a08469050edc1e5da6e24f783162bebc15cc006f717097e56e07fa0b02207c7f47d113355b095c7edd2408d0554aa6b87187bf95652c516fc7747720fb19012102a8d6433a8f799a13d3495f4aa5b99bd288adca5f59e0c9609f15a221220241c5feffffff01f2c49a3b000000001976a914e54445646929fac8b7d6c71715913af44324978488ac00000000e4010061d48e8bd82f0fe9dba3413753c90b695ae75dccee5b3401e76df29b9d33a13900000efda51589f86e30cc2305e7388c01ce0309c19a182cf37bced97c7da72236f660c0a395e765e6e06962ecff5a69d7de359c348a574176c210c37a25d4ffd917866fb0a31976a914e54445646929fac8b7d6c71715913af44324978488aca8141db635af714f32c7415b298937e171c6bb8a3c8627852ae2b69917ece86a411fa51861025bb5ff4ba4ad3be6090c4ba76b1671d70799ed6882c57bcfeaf27cbd558068e9b5fc1553abce8822fcee63e8f6fbb06ad5b753d47e794bfacadbde3f"), SER_DISK, CLIENT_VERSION);
     251           1 :     CTransaction proupregtx(deserialize, stream);
     252             :     // ProUpRevTx from testnet (txid: 70c41a5fea07a80c1297771c43eb58b8f52222b971f61f5c7c2030ec0baf7f9a)
     253           1 :     stream = CDataStream(ParseHex("0300040001b349a0be3ed3bd2513b14589d23b32d68ab372aa8bea9178d5a9d1454ebf3730000000006a4730440220428d87daf3b0ddca9656d59592000f9930a88162c0b017ca460c69c33b76bda202200276a521d94acc57e010262aa1d7d016cebbe7a9806d26dbebb69102ce9cf7b2012102a8d6433a8f799a13d3495f4aa5b99bd288adca5f59e0c9609f15a221220241c5feffffff0183c39a3b000000001976a914e54445646929fac8b7d6c71715913af44324978488ac00000000a4010061d48e8bd82f0fe9dba3413753c90b695ae75dccee5b3401e76df29b9d33a1390000c024ffd34d15686f7a376ecdf45800ddcebf4eb966fd3123cf9450515a38881a0de852c97297bbd9cedb79280f3f1e31b092904a8870274a5a0e52de2ae1dc0bad74851dc0a1e4fe4c274535a2422d3e1510a7be2fbecaae73128e8eb2f336382fb376d2f82273a72960980972b02e3ecf895c00187e0cbe735dc44cb3d97711"), SER_DISK, CLIENT_VERSION);
     254           1 :     CTransaction prouprevtx(deserialize, stream);
     255             : 
     256             :     // if ProRegTx matches, all related pro txes match too
     257           1 :     CBloomFilter filter(10, 0.000001, 0, BLOOM_UPDATE_ALL);
     258           1 :     filter.insert(COutPoint(uint256S("0x23464abc2f724de235e69e72ef5068f1b2701521b88e7b2740b93978ff54909b"), 1));
     259           1 :     BOOST_CHECK_MESSAGE(!filter.IsRelevantAndUpdate(proupservtx), "Bloom filter matches without update");
     260           1 :     BOOST_CHECK_MESSAGE(filter.IsRelevantAndUpdate(proregtx), "Bloom filter didn't match collateral outpoint in ProRegTx");
     261           1 :     BOOST_CHECK_MESSAGE(filter.IsRelevantAndUpdate(proupservtx), "Bloom filter wasn't updated with proregtx hash");
     262             : 
     263             :     // if ProUpRegTx matches, all related pro txes match too
     264           1 :     filter = CBloomFilter(10, 0.000001, 0, BLOOM_UPDATE_ALL);
     265           1 :     filter.insert(ParseHex("359c348a574176c210c37a25d4ffd917866fb0a3"));
     266           1 :     BOOST_CHECK_MESSAGE(!filter.IsRelevantAndUpdate(proupservtx), "Bloom filter matches without update");
     267           1 :     BOOST_CHECK_MESSAGE(filter.IsRelevantAndUpdate(proupregtx), "Bloom filter didn't match Voting keyid in ProUpRegTx");
     268           1 :     BOOST_CHECK_MESSAGE(filter.IsRelevantAndUpdate(proupservtx), "Bloom filter wasn't updated with proregtx hash");
     269           1 : }
     270             : 
     271         149 : BOOST_AUTO_TEST_CASE(merkle_block_1)
     272             : {
     273           1 :     CBlock block = getBlock13b8a();
     274           1 :     CBloomFilter filter(10, 0.000001, 0, BLOOM_UPDATE_ALL);
     275             :     // Match the last transaction
     276           1 :     filter.insert(uint256S("0x74d681e0e03bafa802c8aa084379aa98d9fcd632ddc2ed9782b586ec87451f20"));
     277             : 
     278           1 :     CMerkleBlock merkleBlock(block, filter);
     279           1 :     BOOST_CHECK_EQUAL(merkleBlock.header.GetHash().GetHex(), block.GetHash().GetHex());
     280             : 
     281           1 :     BOOST_CHECK_EQUAL(merkleBlock.vMatchedTxn.size(), 1U);
     282           1 :     std::pair<unsigned int, uint256> pair = merkleBlock.vMatchedTxn[0];
     283             : 
     284           1 :     BOOST_CHECK(merkleBlock.vMatchedTxn[0].second == uint256S("0x74d681e0e03bafa802c8aa084379aa98d9fcd632ddc2ed9782b586ec87451f20"));
     285           1 :     BOOST_CHECK(merkleBlock.vMatchedTxn[0].first == 8);
     286             : 
     287           1 :     std::vector<uint256> vMatched;
     288           1 :     std::vector<unsigned int> vIndex;
     289           1 :     BOOST_CHECK(merkleBlock.txn.ExtractMatches(vMatched, vIndex) == block.hashMerkleRoot);
     290           1 :     BOOST_CHECK(vMatched.size() == merkleBlock.vMatchedTxn.size());
     291           2 :     for (unsigned int i = 0; i < vMatched.size(); i++)
     292           1 :         BOOST_CHECK(vMatched[i] == merkleBlock.vMatchedTxn[i].second);
     293             : 
     294             :     // Also match the 8th transaction
     295           1 :     filter.insert(uint256S("0xdd1fd2a6fc16404faf339881a90adbde7f4f728691ac62e8f168809cdfae1053"));
     296           1 :     merkleBlock = CMerkleBlock(block, filter);
     297           1 :     BOOST_CHECK(merkleBlock.header.GetHash() == block.GetHash());
     298             : 
     299           1 :     BOOST_CHECK(merkleBlock.vMatchedTxn.size() == 2);
     300             : 
     301           1 :     BOOST_CHECK(merkleBlock.vMatchedTxn[1] == pair);
     302             : 
     303           1 :     BOOST_CHECK(merkleBlock.vMatchedTxn[0].second == uint256S("0xdd1fd2a6fc16404faf339881a90adbde7f4f728691ac62e8f168809cdfae1053"));
     304           1 :     BOOST_CHECK(merkleBlock.vMatchedTxn[0].first == 7);
     305             : 
     306           1 :     BOOST_CHECK(merkleBlock.txn.ExtractMatches(vMatched, vIndex) == block.hashMerkleRoot);
     307           1 :     BOOST_CHECK(vMatched.size() == merkleBlock.vMatchedTxn.size());
     308           3 :     for (unsigned int i = 0; i < vMatched.size(); i++)
     309           2 :         BOOST_CHECK(vMatched[i] == merkleBlock.vMatchedTxn[i].second);
     310           1 : }
     311             : 
     312         149 : BOOST_AUTO_TEST_CASE(merkle_block_2)
     313             : {
     314             :     // Random real block (000000005a4ded781e667e06ceefafb71410b511fe0d5adc3e5a27ecbec34ae6)
     315             :     // With 4 txes
     316           1 :     CBlock block;
     317           1 :     CDataStream stream(ParseHex("0100000075616236cc2126035fadb38deb65b9102cc2c41c09cdf29fc051906800000000fe7d5e12ef0ff901f6050211249919b1c0653771832b3a80c66cea42847f0ae1d4d26e49ffff001d00f0a4410401000000010000000000000000000000000000000000000000000000000000000000000000ffffffff0804ffff001d029105ffffffff0100f2052a010000004341046d8709a041d34357697dfcb30a9d05900a6294078012bf3bb09c6f9b525f1d16d5503d7905db1ada9501446ea00728668fc5719aa80be2fdfc8a858a4dbdd4fbac00000000010000000255605dc6f5c3dc148b6da58442b0b2cd422be385eab2ebea4119ee9c268d28350000000049483045022100aa46504baa86df8a33b1192b1b9367b4d729dc41e389f2c04f3e5c7f0559aae702205e82253a54bf5c4f65b7428551554b2045167d6d206dfe6a2e198127d3f7df1501ffffffff55605dc6f5c3dc148b6da58442b0b2cd422be385eab2ebea4119ee9c268d2835010000004847304402202329484c35fa9d6bb32a55a70c0982f606ce0e3634b69006138683bcd12cbb6602200c28feb1e2555c3210f1dddb299738b4ff8bbe9667b68cb8764b5ac17b7adf0001ffffffff0200e1f505000000004341046a0765b5865641ce08dd39690aade26dfbf5511430ca428a3089261361cef170e3929a68aee3d8d4848b0c5111b0a37b82b86ad559fd2a745b44d8e8d9dfdc0cac00180d8f000000004341044a656f065871a353f216ca26cef8dde2f03e8c16202d2e8ad769f02032cb86a5eb5e56842e92e19141d60a01928f8dd2c875a390f67c1f6c94cfc617c0ea45afac0000000001000000025f9a06d3acdceb56be1bfeaa3e8a25e62d182fa24fefe899d1c17f1dad4c2028000000004847304402205d6058484157235b06028c30736c15613a28bdb768ee628094ca8b0030d4d6eb0220328789c9a2ec27ddaec0ad5ef58efded42e6ea17c2e1ce838f3d6913f5e95db601ffffffff5f9a06d3acdceb56be1bfeaa3e8a25e62d182fa24fefe899d1c17f1dad4c2028010000004a493046022100c45af050d3cea806cedd0ab22520c53ebe63b987b8954146cdca42487b84bdd6022100b9b027716a6b59e640da50a864d6dd8a0ef24c76ce62391fa3eabaf4d2886d2d01ffffffff0200e1f505000000004341046a0765b5865641ce08dd39690aade26dfbf5511430ca428a3089261361cef170e3929a68aee3d8d4848b0c5111b0a37b82b86ad559fd2a745b44d8e8d9dfdc0cac00180d8f000000004341046a0765b5865641ce08dd39690aade26dfbf5511430ca428a3089261361cef170e3929a68aee3d8d4848b0c5111b0a37b82b86ad559fd2a745b44d8e8d9dfdc0cac000000000100000002e2274e5fea1bf29d963914bd301aa63b64daaf8a3e88f119b5046ca5738a0f6b0000000048473044022016e7a727a061ea2254a6c358376aaa617ac537eb836c77d646ebda4c748aac8b0220192ce28bf9f2c06a6467e6531e27648d2b3e2e2bae85159c9242939840295ba501ffffffffe2274e5fea1bf29d963914bd301aa63b64daaf8a3e88f119b5046ca5738a0f6b010000004a493046022100b7a1a755588d4190118936e15cd217d133b0e4a53c3c15924010d5648d8925c9022100aaef031874db2114f2d869ac2de4ae53908fbfea5b2b1862e181626bb9005c9f01ffffffff0200e1f505000000004341044a656f065871a353f216ca26cef8dde2f03e8c16202d2e8ad769f02032cb86a5eb5e56842e92e19141d60a01928f8dd2c875a390f67c1f6c94cfc617c0ea45afac00180d8f000000004341046a0765b5865641ce08dd39690aade26dfbf5511430ca428a3089261361cef170e3929a68aee3d8d4848b0c5111b0a37b82b86ad559fd2a745b44d8e8d9dfdc0cac00000000"), SER_NETWORK, PROTOCOL_VERSION);
     318           1 :     stream >> block;
     319             : 
     320           1 :     CBloomFilter filter(10, 0.000001, 0, BLOOM_UPDATE_ALL);
     321             :     // Match the first transaction
     322           1 :     filter.insert(uint256S("0xe980fe9f792d014e73b95203dc1335c5f9ce19ac537a419e6df5b47aecb93b70"));
     323             : 
     324           1 :     CMerkleBlock merkleBlock(block, filter);
     325           1 :     BOOST_CHECK(merkleBlock.header.GetHash() == block.GetHash());
     326             : 
     327           1 :     BOOST_CHECK(merkleBlock.vMatchedTxn.size() == 1);
     328           1 :     std::pair<unsigned int, uint256> pair = merkleBlock.vMatchedTxn[0];
     329             : 
     330           1 :     BOOST_CHECK(merkleBlock.vMatchedTxn[0].second == uint256S("0xe980fe9f792d014e73b95203dc1335c5f9ce19ac537a419e6df5b47aecb93b70"));
     331           1 :     BOOST_CHECK(merkleBlock.vMatchedTxn[0].first == 0);
     332             : 
     333           1 :     std::vector<uint256> vMatched;
     334           1 :     std::vector<unsigned int> vIndex;
     335           1 :     BOOST_CHECK(merkleBlock.txn.ExtractMatches(vMatched, vIndex) == block.hashMerkleRoot);
     336           1 :     BOOST_CHECK(vMatched.size() == merkleBlock.vMatchedTxn.size());
     337           2 :     for (unsigned int i = 0; i < vMatched.size(); i++)
     338           1 :         BOOST_CHECK(vMatched[i] == merkleBlock.vMatchedTxn[i].second);
     339             : 
     340             :     // Match an output from the second transaction (the pubkey for address 1DZTzaBHUDM7T3QvUKBz4qXMRpkg8jsfB5)
     341             :     // This should match the third transaction because it spends the output matched
     342             :     // It also matches the fourth transaction, which spends to the pubkey again
     343           1 :     filter.insert(ParseHex("044a656f065871a353f216ca26cef8dde2f03e8c16202d2e8ad769f02032cb86a5eb5e56842e92e19141d60a01928f8dd2c875a390f67c1f6c94cfc617c0ea45af"));
     344             : 
     345           1 :     merkleBlock = CMerkleBlock(block, filter);
     346           1 :     BOOST_CHECK(merkleBlock.header.GetHash() == block.GetHash());
     347             : 
     348           1 :     BOOST_CHECK(merkleBlock.vMatchedTxn.size() == 4);
     349             : 
     350           1 :     BOOST_CHECK(pair == merkleBlock.vMatchedTxn[0]);
     351             : 
     352           1 :     BOOST_CHECK(merkleBlock.vMatchedTxn[1].second == uint256S("0x28204cad1d7fc1d199e8ef4fa22f182de6258a3eaafe1bbe56ebdcacd3069a5f"));
     353           1 :     BOOST_CHECK(merkleBlock.vMatchedTxn[1].first == 1);
     354             : 
     355           1 :     BOOST_CHECK(merkleBlock.vMatchedTxn[2].second == uint256S("0x6b0f8a73a56c04b519f1883e8aafda643ba61a30bd1439969df21bea5f4e27e2"));
     356           1 :     BOOST_CHECK(merkleBlock.vMatchedTxn[2].first == 2);
     357             : 
     358           1 :     BOOST_CHECK(merkleBlock.vMatchedTxn[3].second == uint256S("0x3c1d7e82342158e4109df2e0b6348b6e84e403d8b4046d7007663ace63cddb23"));
     359           1 :     BOOST_CHECK(merkleBlock.vMatchedTxn[3].first == 3);
     360             : 
     361           1 :     BOOST_CHECK(merkleBlock.txn.ExtractMatches(vMatched, vIndex) == block.hashMerkleRoot);
     362           1 :     BOOST_CHECK(vMatched.size() == merkleBlock.vMatchedTxn.size());
     363           5 :     for (unsigned int i = 0; i < vMatched.size(); i++)
     364           4 :         BOOST_CHECK(vMatched[i] == merkleBlock.vMatchedTxn[i].second);
     365           1 : }
     366             : 
     367         149 : BOOST_AUTO_TEST_CASE(merkle_block_2_with_update_none)
     368             : {
     369             :     // Random real block (000000005a4ded781e667e06ceefafb71410b511fe0d5adc3e5a27ecbec34ae6)
     370             :     // With 4 txes
     371           1 :     CBlock block;
     372           1 :     CDataStream stream(ParseHex("0100000075616236cc2126035fadb38deb65b9102cc2c41c09cdf29fc051906800000000fe7d5e12ef0ff901f6050211249919b1c0653771832b3a80c66cea42847f0ae1d4d26e49ffff001d00f0a4410401000000010000000000000000000000000000000000000000000000000000000000000000ffffffff0804ffff001d029105ffffffff0100f2052a010000004341046d8709a041d34357697dfcb30a9d05900a6294078012bf3bb09c6f9b525f1d16d5503d7905db1ada9501446ea00728668fc5719aa80be2fdfc8a858a4dbdd4fbac00000000010000000255605dc6f5c3dc148b6da58442b0b2cd422be385eab2ebea4119ee9c268d28350000000049483045022100aa46504baa86df8a33b1192b1b9367b4d729dc41e389f2c04f3e5c7f0559aae702205e82253a54bf5c4f65b7428551554b2045167d6d206dfe6a2e198127d3f7df1501ffffffff55605dc6f5c3dc148b6da58442b0b2cd422be385eab2ebea4119ee9c268d2835010000004847304402202329484c35fa9d6bb32a55a70c0982f606ce0e3634b69006138683bcd12cbb6602200c28feb1e2555c3210f1dddb299738b4ff8bbe9667b68cb8764b5ac17b7adf0001ffffffff0200e1f505000000004341046a0765b5865641ce08dd39690aade26dfbf5511430ca428a3089261361cef170e3929a68aee3d8d4848b0c5111b0a37b82b86ad559fd2a745b44d8e8d9dfdc0cac00180d8f000000004341044a656f065871a353f216ca26cef8dde2f03e8c16202d2e8ad769f02032cb86a5eb5e56842e92e19141d60a01928f8dd2c875a390f67c1f6c94cfc617c0ea45afac0000000001000000025f9a06d3acdceb56be1bfeaa3e8a25e62d182fa24fefe899d1c17f1dad4c2028000000004847304402205d6058484157235b06028c30736c15613a28bdb768ee628094ca8b0030d4d6eb0220328789c9a2ec27ddaec0ad5ef58efded42e6ea17c2e1ce838f3d6913f5e95db601ffffffff5f9a06d3acdceb56be1bfeaa3e8a25e62d182fa24fefe899d1c17f1dad4c2028010000004a493046022100c45af050d3cea806cedd0ab22520c53ebe63b987b8954146cdca42487b84bdd6022100b9b027716a6b59e640da50a864d6dd8a0ef24c76ce62391fa3eabaf4d2886d2d01ffffffff0200e1f505000000004341046a0765b5865641ce08dd39690aade26dfbf5511430ca428a3089261361cef170e3929a68aee3d8d4848b0c5111b0a37b82b86ad559fd2a745b44d8e8d9dfdc0cac00180d8f000000004341046a0765b5865641ce08dd39690aade26dfbf5511430ca428a3089261361cef170e3929a68aee3d8d4848b0c5111b0a37b82b86ad559fd2a745b44d8e8d9dfdc0cac000000000100000002e2274e5fea1bf29d963914bd301aa63b64daaf8a3e88f119b5046ca5738a0f6b0000000048473044022016e7a727a061ea2254a6c358376aaa617ac537eb836c77d646ebda4c748aac8b0220192ce28bf9f2c06a6467e6531e27648d2b3e2e2bae85159c9242939840295ba501ffffffffe2274e5fea1bf29d963914bd301aa63b64daaf8a3e88f119b5046ca5738a0f6b010000004a493046022100b7a1a755588d4190118936e15cd217d133b0e4a53c3c15924010d5648d8925c9022100aaef031874db2114f2d869ac2de4ae53908fbfea5b2b1862e181626bb9005c9f01ffffffff0200e1f505000000004341044a656f065871a353f216ca26cef8dde2f03e8c16202d2e8ad769f02032cb86a5eb5e56842e92e19141d60a01928f8dd2c875a390f67c1f6c94cfc617c0ea45afac00180d8f000000004341046a0765b5865641ce08dd39690aade26dfbf5511430ca428a3089261361cef170e3929a68aee3d8d4848b0c5111b0a37b82b86ad559fd2a745b44d8e8d9dfdc0cac00000000"), SER_NETWORK, PROTOCOL_VERSION);
     373           1 :     stream >> block;
     374             : 
     375           1 :     CBloomFilter filter(10, 0.000001, 0, BLOOM_UPDATE_NONE);
     376             :     // Match the first transaction
     377           1 :     filter.insert(uint256S("0xe980fe9f792d014e73b95203dc1335c5f9ce19ac537a419e6df5b47aecb93b70"));
     378             : 
     379           1 :     CMerkleBlock merkleBlock(block, filter);
     380           1 :     BOOST_CHECK(merkleBlock.header.GetHash() == block.GetHash());
     381             : 
     382           1 :     BOOST_CHECK(merkleBlock.vMatchedTxn.size() == 1);
     383           1 :     std::pair<unsigned int, uint256> pair = merkleBlock.vMatchedTxn[0];
     384             : 
     385           1 :     BOOST_CHECK(merkleBlock.vMatchedTxn[0].second == uint256S("0xe980fe9f792d014e73b95203dc1335c5f9ce19ac537a419e6df5b47aecb93b70"));
     386           1 :     BOOST_CHECK(merkleBlock.vMatchedTxn[0].first == 0);
     387             : 
     388           1 :     std::vector<uint256> vMatched;
     389           1 :     std::vector<unsigned int> vIndex;
     390           1 :     BOOST_CHECK(merkleBlock.txn.ExtractMatches(vMatched, vIndex) == block.hashMerkleRoot);
     391           1 :     BOOST_CHECK(vMatched.size() == merkleBlock.vMatchedTxn.size());
     392           2 :     for (unsigned int i = 0; i < vMatched.size(); i++)
     393           1 :         BOOST_CHECK(vMatched[i] == merkleBlock.vMatchedTxn[i].second);
     394             : 
     395             :     // Match an output from the second transaction (the pubkey for address 1DZTzaBHUDM7T3QvUKBz4qXMRpkg8jsfB5)
     396             :     // This should not match the third transaction though it spends the output matched
     397             :     // It will match the fourth transaction, which has another pay-to-pubkey output to the same address
     398           1 :     filter.insert(ParseHex("044a656f065871a353f216ca26cef8dde2f03e8c16202d2e8ad769f02032cb86a5eb5e56842e92e19141d60a01928f8dd2c875a390f67c1f6c94cfc617c0ea45af"));
     399             : 
     400           1 :     merkleBlock = CMerkleBlock(block, filter);
     401           1 :     BOOST_CHECK(merkleBlock.header.GetHash() == block.GetHash());
     402             : 
     403           1 :     BOOST_CHECK(merkleBlock.vMatchedTxn.size() == 3);
     404             : 
     405           1 :     BOOST_CHECK(pair == merkleBlock.vMatchedTxn[0]);
     406             : 
     407           1 :     BOOST_CHECK(merkleBlock.vMatchedTxn[1].second == uint256S("0x28204cad1d7fc1d199e8ef4fa22f182de6258a3eaafe1bbe56ebdcacd3069a5f"));
     408           1 :     BOOST_CHECK(merkleBlock.vMatchedTxn[1].first == 1);
     409             : 
     410           1 :     BOOST_CHECK(merkleBlock.vMatchedTxn[2].second == uint256S("0x3c1d7e82342158e4109df2e0b6348b6e84e403d8b4046d7007663ace63cddb23"));
     411           1 :     BOOST_CHECK(merkleBlock.vMatchedTxn[2].first == 3);
     412             : 
     413           1 :     BOOST_CHECK(merkleBlock.txn.ExtractMatches(vMatched, vIndex) == block.hashMerkleRoot);
     414           1 :     BOOST_CHECK(vMatched.size() == merkleBlock.vMatchedTxn.size());
     415           4 :     for (unsigned int i = 0; i < vMatched.size(); i++)
     416           3 :         BOOST_CHECK(vMatched[i] == merkleBlock.vMatchedTxn[i].second);
     417           1 : }
     418             : 
     419         149 : BOOST_AUTO_TEST_CASE(merkle_block_3_and_serialize)
     420             : {
     421             :     // Random real block (000000000000dab0130bbcc991d3d7ae6b81aa6f50a798888dfe62337458dc45)
     422             :     // With one tx
     423           1 :     CBlock block;
     424           1 :     CDataStream stream(ParseHex("0100000079cda856b143d9db2c1caff01d1aecc8630d30625d10e8b4b8b0000000000000b50cc069d6a3e33e3ff84a5c41d9d3febe7c770fdcc96b2c3ff60abe184f196367291b4d4c86041b8fa45d630101000000010000000000000000000000000000000000000000000000000000000000000000ffffffff08044c86041b020a02ffffffff0100f2052a01000000434104ecd3229b0571c3be876feaac0442a9f13c5a572742927af1dc623353ecf8c202225f64868137a18cdd85cbbb4c74fbccfd4f49639cf1bdc94a5672bb15ad5d4cac00000000"), SER_NETWORK, PROTOCOL_VERSION);
     425           1 :     stream >> block;
     426             : 
     427           1 :     CBloomFilter filter(10, 0.000001, 0, BLOOM_UPDATE_ALL);
     428             :     // Match the only transaction
     429           1 :     filter.insert(uint256S("0x63194f18be0af63f2c6bc9dc0f777cbefed3d9415c4af83f3ee3a3d669c00cb5"));
     430             : 
     431           1 :     CMerkleBlock merkleBlock(block, filter);
     432           1 :     BOOST_CHECK(merkleBlock.header.GetHash() == block.GetHash());
     433             : 
     434           1 :     BOOST_CHECK(merkleBlock.vMatchedTxn.size() == 1);
     435             : 
     436           1 :     BOOST_CHECK(merkleBlock.vMatchedTxn[0].second == uint256S("0x63194f18be0af63f2c6bc9dc0f777cbefed3d9415c4af83f3ee3a3d669c00cb5"));
     437           1 :     BOOST_CHECK(merkleBlock.vMatchedTxn[0].first == 0);
     438             : 
     439           1 :     std::vector<uint256> vMatched;
     440           1 :     std::vector<unsigned int> vIndex;
     441           1 :     BOOST_CHECK(merkleBlock.txn.ExtractMatches(vMatched, vIndex) == block.hashMerkleRoot);
     442           1 :     BOOST_CHECK(vMatched.size() == merkleBlock.vMatchedTxn.size());
     443           2 :     for (unsigned int i = 0; i < vMatched.size(); i++)
     444           1 :         BOOST_CHECK(vMatched[i] == merkleBlock.vMatchedTxn[i].second);
     445             : 
     446           1 :     CDataStream merkleStream(SER_NETWORK, PROTOCOL_VERSION);
     447           1 :     merkleStream << merkleBlock;
     448             : 
     449           1 :     std::vector<uint8_t> expected = ParseHex("0100000079cda856b143d9db2c1caff01d1aecc8630d30625d10e8b4b8b0000000000000b50cc069d6a3e33e3ff84a5c41d9d3febe7c770fdcc96b2c3ff60abe184f196367291b4d4c86041b8fa45d630100000001b50cc069d6a3e33e3ff84a5c41d9d3febe7c770fdcc96b2c3ff60abe184f19630101");
     450           1 :     auto result{MakeUCharSpan(merkleStream)};
     451             : 
     452           1 :     BOOST_CHECK_EQUAL_COLLECTIONS(expected.begin(), expected.end(), result.begin(), result.end());
     453           1 : }
     454             : 
     455         149 : BOOST_AUTO_TEST_CASE(merkle_block_4)
     456             : {
     457             :     // Random real block (000000000000b731f2eef9e8c63173adfb07e41bd53eb0ef0a6b720d6cb6dea4)
     458             :     // With 7 txes
     459           1 :     CBlock block;
     460           1 :     CDataStream stream(ParseHex("0100000082bb869cf3a793432a66e826e05a6fc37469f8efb7421dc880670100000000007f16c5962e8bd963659c793ce370d95f093bc7e367117b3c30c1f8fdd0d9728776381b4d4c86041b554b85290701000000010000000000000000000000000000000000000000000000000000000000000000ffffffff07044c86041b0136ffffffff0100f2052a01000000434104eaafc2314def4ca98ac970241bcab022b9c1e1f4ea423a20f134c876f2c01ec0f0dd5b2e86e7168cefe0d81113c3807420ce13ad1357231a2252247d97a46a91ac000000000100000001bcad20a6a29827d1424f08989255120bf7f3e9e3cdaaa6bb31b0737fe048724300000000494830450220356e834b046cadc0f8ebb5a8a017b02de59c86305403dad52cd77b55af062ea10221009253cd6c119d4729b77c978e1e2aa19f5ea6e0e52b3f16e32fa608cd5bab753901ffffffff02008d380c010000001976a9142b4b8072ecbba129b6453c63e129e643207249ca88ac0065cd1d000000001976a9141b8dd13b994bcfc787b32aeadf58ccb3615cbd5488ac000000000100000003fdacf9b3eb077412e7a968d2e4f11b9a9dee312d666187ed77ee7d26af16cb0b000000008c493046022100ea1608e70911ca0de5af51ba57ad23b9a51db8d28f82c53563c56a05c20f5a87022100a8bdc8b4a8acc8634c6b420410150775eb7f2474f5615f7fccd65af30f310fbf01410465fdf49e29b06b9a1582287b6279014f834edc317695d125ef623c1cc3aaece245bd69fcad7508666e9c74a49dc9056d5fc14338ef38118dc4afae5fe2c585caffffffff309e1913634ecb50f3c4f83e96e70b2df071b497b8973a3e75429df397b5af83000000004948304502202bdb79c596a9ffc24e96f4386199aba386e9bc7b6071516e2b51dda942b3a1ed022100c53a857e76b724fc14d45311eac5019650d415c3abb5428f3aae16d8e69bec2301ffffffff2089e33491695080c9edc18a428f7d834db5b6d372df13ce2b1b0e0cbcb1e6c10000000049483045022100d4ce67c5896ee251c810ac1ff9ceccd328b497c8f553ab6e08431e7d40bad6b5022033119c0c2b7d792d31f1187779c7bd95aefd93d90a715586d73801d9b47471c601ffffffff0100714460030000001976a914c7b55141d097ea5df7a0ed330cf794376e53ec8d88ac0000000001000000045bf0e214aa4069a3e792ecee1e1bf0c1d397cde8dd08138f4b72a00681743447000000008b48304502200c45de8c4f3e2c1821f2fc878cba97b1e6f8807d94930713aa1c86a67b9bf1e40221008581abfef2e30f957815fc89978423746b2086375ca8ecf359c85c2a5b7c88ad01410462bb73f76ca0994fcb8b4271e6fb7561f5c0f9ca0cf6485261c4a0dc894f4ab844c6cdfb97cd0b60ffb5018ffd6238f4d87270efb1d3ae37079b794a92d7ec95ffffffffd669f7d7958d40fc59d2253d88e0f248e29b599c80bbcec344a83dda5f9aa72c000000008a473044022078124c8beeaa825f9e0b30bff96e564dd859432f2d0cb3b72d3d5d93d38d7e930220691d233b6c0f995be5acb03d70a7f7a65b6bc9bdd426260f38a1346669507a3601410462bb73f76ca0994fcb8b4271e6fb7561f5c0f9ca0cf6485261c4a0dc894f4ab844c6cdfb97cd0b60ffb5018ffd6238f4d87270efb1d3ae37079b794a92d7ec95fffffffff878af0d93f5229a68166cf051fd372bb7a537232946e0a46f53636b4dafdaa4000000008c493046022100c717d1714551663f69c3c5759bdbb3a0fcd3fab023abc0e522fe6440de35d8290221008d9cbe25bffc44af2b18e81c58eb37293fd7fe1c2e7b46fc37ee8c96c50ab1e201410462bb73f76ca0994fcb8b4271e6fb7561f5c0f9ca0cf6485261c4a0dc894f4ab844c6cdfb97cd0b60ffb5018ffd6238f4d87270efb1d3ae37079b794a92d7ec95ffffffff27f2b668859cd7f2f894aa0fd2d9e60963bcd07c88973f425f999b8cbfd7a1e2000000008c493046022100e00847147cbf517bcc2f502f3ddc6d284358d102ed20d47a8aa788a62f0db780022100d17b2d6fa84dcaf1c95d88d7e7c30385aecf415588d749afd3ec81f6022cecd701410462bb73f76ca0994fcb8b4271e6fb7561f5c0f9ca0cf6485261c4a0dc894f4ab844c6cdfb97cd0b60ffb5018ffd6238f4d87270efb1d3ae37079b794a92d7ec95ffffffff0100c817a8040000001976a914b6efd80d99179f4f4ff6f4dd0a007d018c385d2188ac000000000100000001834537b2f1ce8ef9373a258e10545ce5a50b758df616cd4356e0032554ebd3c4000000008b483045022100e68f422dd7c34fdce11eeb4509ddae38201773dd62f284e8aa9d96f85099d0b002202243bd399ff96b649a0fad05fa759d6a882f0af8c90cf7632c2840c29070aec20141045e58067e815c2f464c6a2a15f987758374203895710c2d452442e28496ff38ba8f5fd901dc20e29e88477167fe4fc299bf818fd0d9e1632d467b2a3d9503b1aaffffffff0280d7e636030000001976a914f34c3e10eb387efe872acb614c89e78bfca7815d88ac404b4c00000000001976a914a84e272933aaf87e1715d7786c51dfaeb5b65a6f88ac00000000010000000143ac81c8e6f6ef307dfe17f3d906d999e23e0189fda838c5510d850927e03ae7000000008c4930460221009c87c344760a64cb8ae6685a3eec2c1ac1bed5b88c87de51acd0e124f266c16602210082d07c037359c3a257b5c63ebd90f5a5edf97b2ac1c434b08ca998839f346dd40141040ba7e521fa7946d12edbb1d1e95a15c34bd4398195e86433c92b431cd315f455fe30032ede69cad9d1e1ed6c3c4ec0dbfced53438c625462afb792dcb098544bffffffff0240420f00000000001976a9144676d1b820d63ec272f1900d59d43bc6463d96f888ac40420f00000000001976a914648d04341d00d7968b3405c034adc38d4d8fb9bd88ac00000000010000000248cc917501ea5c55f4a8d2009c0567c40cfe037c2e71af017d0a452ff705e3f1000000008b483045022100bf5fdc86dc5f08a5d5c8e43a8c9d5b1ed8c65562e280007b52b133021acd9acc02205e325d613e555f772802bf413d36ba807892ed1a690a77811d3033b3de226e0a01410429fa713b124484cb2bd7b5557b2c0b9df7b2b1fee61825eadc5ae6c37a9920d38bfccdc7dc3cb0c47d7b173dbc9db8d37db0a33ae487982c59c6f8606e9d1791ffffffff41ed70551dd7e841883ab8f0b16bf04176b7d1480e4f0af9f3d4c3595768d068000000008b4830450221008513ad65187b903aed1102d1d0c47688127658c51106753fed0151ce9c16b80902201432b9ebcb87bd04ceb2de66035fbbaf4bf8b00d1cfe41f1a1f7338f9ad79d210141049d4cf80125bf50be1709f718c07ad15d0fc612b7da1f5570dddc35f2a352f0f27c978b06820edca9ef982c35fda2d255afba340068c5035552368bc7200c1488ffffffff0100093d00000000001976a9148edb68822f1ad580b043c7b3df2e400f8699eb4888ac00000000"), SER_NETWORK, PROTOCOL_VERSION);
     461           1 :     stream >> block;
     462             : 
     463           1 :     CBloomFilter filter(10, 0.000001, 0, BLOOM_UPDATE_ALL);
     464             :     // Match the last transaction
     465           1 :     filter.insert(uint256S("0x0a2a92f0bda4727d0a13eaddf4dd9ac6b5c61a1429e6b2b818f19b15df0ac154"));
     466             : 
     467           1 :     CMerkleBlock merkleBlock(block, filter);
     468           1 :     BOOST_CHECK(merkleBlock.header.GetHash() == block.GetHash());
     469             : 
     470           1 :     BOOST_CHECK(merkleBlock.vMatchedTxn.size() == 1);
     471           1 :     std::pair<unsigned int, uint256> pair = merkleBlock.vMatchedTxn[0];
     472             : 
     473           1 :     BOOST_CHECK(merkleBlock.vMatchedTxn[0].second == uint256S("0x0a2a92f0bda4727d0a13eaddf4dd9ac6b5c61a1429e6b2b818f19b15df0ac154"));
     474           1 :     BOOST_CHECK(merkleBlock.vMatchedTxn[0].first == 6);
     475             : 
     476           1 :     std::vector<uint256> vMatched;
     477           1 :     std::vector<unsigned int> vIndex;
     478           1 :     BOOST_CHECK(merkleBlock.txn.ExtractMatches(vMatched, vIndex) == block.hashMerkleRoot);
     479           1 :     BOOST_CHECK(vMatched.size() == merkleBlock.vMatchedTxn.size());
     480           2 :     for (unsigned int i = 0; i < vMatched.size(); i++)
     481           1 :         BOOST_CHECK(vMatched[i] == merkleBlock.vMatchedTxn[i].second);
     482             : 
     483             :     // Also match the 4th transaction
     484           1 :     filter.insert(uint256S("0x02981fa052f0481dbc5868f4fc2166035a10f27a03cfd2de67326471df5bc041"));
     485           1 :     merkleBlock = CMerkleBlock(block, filter);
     486           1 :     BOOST_CHECK(merkleBlock.header.GetHash() == block.GetHash());
     487             : 
     488           1 :     BOOST_CHECK(merkleBlock.vMatchedTxn.size() == 2);
     489             : 
     490           1 :     BOOST_CHECK(merkleBlock.vMatchedTxn[0].second == uint256S("0x02981fa052f0481dbc5868f4fc2166035a10f27a03cfd2de67326471df5bc041"));
     491           1 :     BOOST_CHECK(merkleBlock.vMatchedTxn[0].first == 3);
     492             : 
     493           1 :     BOOST_CHECK(merkleBlock.vMatchedTxn[1] == pair);
     494             : 
     495           1 :     BOOST_CHECK(merkleBlock.txn.ExtractMatches(vMatched, vIndex) == block.hashMerkleRoot);
     496           1 :     BOOST_CHECK(vMatched.size() == merkleBlock.vMatchedTxn.size());
     497           3 :     for (unsigned int i = 0; i < vMatched.size(); i++)
     498           2 :         BOOST_CHECK(vMatched[i] == merkleBlock.vMatchedTxn[i].second);
     499           1 : }
     500             : 
     501         149 : BOOST_AUTO_TEST_CASE(merkle_block_4_test_p2pubkey_only)
     502             : {
     503             :     // Random real block (000000000000b731f2eef9e8c63173adfb07e41bd53eb0ef0a6b720d6cb6dea4)
     504             :     // With 7 txes
     505           1 :     CBlock block;
     506           1 :     CDataStream stream(ParseHex("0100000082bb869cf3a793432a66e826e05a6fc37469f8efb7421dc880670100000000007f16c5962e8bd963659c793ce370d95f093bc7e367117b3c30c1f8fdd0d9728776381b4d4c86041b554b85290701000000010000000000000000000000000000000000000000000000000000000000000000ffffffff07044c86041b0136ffffffff0100f2052a01000000434104eaafc2314def4ca98ac970241bcab022b9c1e1f4ea423a20f134c876f2c01ec0f0dd5b2e86e7168cefe0d81113c3807420ce13ad1357231a2252247d97a46a91ac000000000100000001bcad20a6a29827d1424f08989255120bf7f3e9e3cdaaa6bb31b0737fe048724300000000494830450220356e834b046cadc0f8ebb5a8a017b02de59c86305403dad52cd77b55af062ea10221009253cd6c119d4729b77c978e1e2aa19f5ea6e0e52b3f16e32fa608cd5bab753901ffffffff02008d380c010000001976a9142b4b8072ecbba129b6453c63e129e643207249ca88ac0065cd1d000000001976a9141b8dd13b994bcfc787b32aeadf58ccb3615cbd5488ac000000000100000003fdacf9b3eb077412e7a968d2e4f11b9a9dee312d666187ed77ee7d26af16cb0b000000008c493046022100ea1608e70911ca0de5af51ba57ad23b9a51db8d28f82c53563c56a05c20f5a87022100a8bdc8b4a8acc8634c6b420410150775eb7f2474f5615f7fccd65af30f310fbf01410465fdf49e29b06b9a1582287b6279014f834edc317695d125ef623c1cc3aaece245bd69fcad7508666e9c74a49dc9056d5fc14338ef38118dc4afae5fe2c585caffffffff309e1913634ecb50f3c4f83e96e70b2df071b497b8973a3e75429df397b5af83000000004948304502202bdb79c596a9ffc24e96f4386199aba386e9bc7b6071516e2b51dda942b3a1ed022100c53a857e76b724fc14d45311eac5019650d415c3abb5428f3aae16d8e69bec2301ffffffff2089e33491695080c9edc18a428f7d834db5b6d372df13ce2b1b0e0cbcb1e6c10000000049483045022100d4ce67c5896ee251c810ac1ff9ceccd328b497c8f553ab6e08431e7d40bad6b5022033119c0c2b7d792d31f1187779c7bd95aefd93d90a715586d73801d9b47471c601ffffffff0100714460030000001976a914c7b55141d097ea5df7a0ed330cf794376e53ec8d88ac0000000001000000045bf0e214aa4069a3e792ecee1e1bf0c1d397cde8dd08138f4b72a00681743447000000008b48304502200c45de8c4f3e2c1821f2fc878cba97b1e6f8807d94930713aa1c86a67b9bf1e40221008581abfef2e30f957815fc89978423746b2086375ca8ecf359c85c2a5b7c88ad01410462bb73f76ca0994fcb8b4271e6fb7561f5c0f9ca0cf6485261c4a0dc894f4ab844c6cdfb97cd0b60ffb5018ffd6238f4d87270efb1d3ae37079b794a92d7ec95ffffffffd669f7d7958d40fc59d2253d88e0f248e29b599c80bbcec344a83dda5f9aa72c000000008a473044022078124c8beeaa825f9e0b30bff96e564dd859432f2d0cb3b72d3d5d93d38d7e930220691d233b6c0f995be5acb03d70a7f7a65b6bc9bdd426260f38a1346669507a3601410462bb73f76ca0994fcb8b4271e6fb7561f5c0f9ca0cf6485261c4a0dc894f4ab844c6cdfb97cd0b60ffb5018ffd6238f4d87270efb1d3ae37079b794a92d7ec95fffffffff878af0d93f5229a68166cf051fd372bb7a537232946e0a46f53636b4dafdaa4000000008c493046022100c717d1714551663f69c3c5759bdbb3a0fcd3fab023abc0e522fe6440de35d8290221008d9cbe25bffc44af2b18e81c58eb37293fd7fe1c2e7b46fc37ee8c96c50ab1e201410462bb73f76ca0994fcb8b4271e6fb7561f5c0f9ca0cf6485261c4a0dc894f4ab844c6cdfb97cd0b60ffb5018ffd6238f4d87270efb1d3ae37079b794a92d7ec95ffffffff27f2b668859cd7f2f894aa0fd2d9e60963bcd07c88973f425f999b8cbfd7a1e2000000008c493046022100e00847147cbf517bcc2f502f3ddc6d284358d102ed20d47a8aa788a62f0db780022100d17b2d6fa84dcaf1c95d88d7e7c30385aecf415588d749afd3ec81f6022cecd701410462bb73f76ca0994fcb8b4271e6fb7561f5c0f9ca0cf6485261c4a0dc894f4ab844c6cdfb97cd0b60ffb5018ffd6238f4d87270efb1d3ae37079b794a92d7ec95ffffffff0100c817a8040000001976a914b6efd80d99179f4f4ff6f4dd0a007d018c385d2188ac000000000100000001834537b2f1ce8ef9373a258e10545ce5a50b758df616cd4356e0032554ebd3c4000000008b483045022100e68f422dd7c34fdce11eeb4509ddae38201773dd62f284e8aa9d96f85099d0b002202243bd399ff96b649a0fad05fa759d6a882f0af8c90cf7632c2840c29070aec20141045e58067e815c2f464c6a2a15f987758374203895710c2d452442e28496ff38ba8f5fd901dc20e29e88477167fe4fc299bf818fd0d9e1632d467b2a3d9503b1aaffffffff0280d7e636030000001976a914f34c3e10eb387efe872acb614c89e78bfca7815d88ac404b4c00000000001976a914a84e272933aaf87e1715d7786c51dfaeb5b65a6f88ac00000000010000000143ac81c8e6f6ef307dfe17f3d906d999e23e0189fda838c5510d850927e03ae7000000008c4930460221009c87c344760a64cb8ae6685a3eec2c1ac1bed5b88c87de51acd0e124f266c16602210082d07c037359c3a257b5c63ebd90f5a5edf97b2ac1c434b08ca998839f346dd40141040ba7e521fa7946d12edbb1d1e95a15c34bd4398195e86433c92b431cd315f455fe30032ede69cad9d1e1ed6c3c4ec0dbfced53438c625462afb792dcb098544bffffffff0240420f00000000001976a9144676d1b820d63ec272f1900d59d43bc6463d96f888ac40420f00000000001976a914648d04341d00d7968b3405c034adc38d4d8fb9bd88ac00000000010000000248cc917501ea5c55f4a8d2009c0567c40cfe037c2e71af017d0a452ff705e3f1000000008b483045022100bf5fdc86dc5f08a5d5c8e43a8c9d5b1ed8c65562e280007b52b133021acd9acc02205e325d613e555f772802bf413d36ba807892ed1a690a77811d3033b3de226e0a01410429fa713b124484cb2bd7b5557b2c0b9df7b2b1fee61825eadc5ae6c37a9920d38bfccdc7dc3cb0c47d7b173dbc9db8d37db0a33ae487982c59c6f8606e9d1791ffffffff41ed70551dd7e841883ab8f0b16bf04176b7d1480e4f0af9f3d4c3595768d068000000008b4830450221008513ad65187b903aed1102d1d0c47688127658c51106753fed0151ce9c16b80902201432b9ebcb87bd04ceb2de66035fbbaf4bf8b00d1cfe41f1a1f7338f9ad79d210141049d4cf80125bf50be1709f718c07ad15d0fc612b7da1f5570dddc35f2a352f0f27c978b06820edca9ef982c35fda2d255afba340068c5035552368bc7200c1488ffffffff0100093d00000000001976a9148edb68822f1ad580b043c7b3df2e400f8699eb4888ac00000000"), SER_NETWORK, PROTOCOL_VERSION);
     507           1 :     stream >> block;
     508             : 
     509           1 :     CBloomFilter filter(10, 0.000001, 0, BLOOM_UPDATE_P2PUBKEY_ONLY);
     510             :     // Match the generation pubkey
     511           1 :     filter.insert(ParseHex("04eaafc2314def4ca98ac970241bcab022b9c1e1f4ea423a20f134c876f2c01ec0f0dd5b2e86e7168cefe0d81113c3807420ce13ad1357231a2252247d97a46a91"));
     512             :     // ...and the output address of the 4th transaction
     513           1 :     filter.insert(ParseHex("b6efd80d99179f4f4ff6f4dd0a007d018c385d21"));
     514             : 
     515           1 :     CMerkleBlock merkleBlock(block, filter);
     516           1 :     BOOST_CHECK(merkleBlock.header.GetHash() == block.GetHash());
     517             : 
     518             :     // We should match the generation outpoint
     519           1 :     BOOST_CHECK(filter.contains(COutPoint(uint256S("0x147caa76786596590baa4e98f5d9f48b86c7765e489f7a6ff3360fe5c674360b"), 0)));
     520             :     // ... but not the 4th transaction's output (its not pay-2-pubkey)
     521           1 :     BOOST_CHECK(!filter.contains(COutPoint(uint256S("0x02981fa052f0481dbc5868f4fc2166035a10f27a03cfd2de67326471df5bc041"), 0)));
     522           1 : }
     523             : 
     524         149 : BOOST_AUTO_TEST_CASE(merkle_block_4_test_update_none)
     525             : {
     526             :     // Random real block (000000000000b731f2eef9e8c63173adfb07e41bd53eb0ef0a6b720d6cb6dea4)
     527             :     // With 7 txes
     528           1 :     CBlock block;
     529           1 :     CDataStream stream(ParseHex("0100000082bb869cf3a793432a66e826e05a6fc37469f8efb7421dc880670100000000007f16c5962e8bd963659c793ce370d95f093bc7e367117b3c30c1f8fdd0d9728776381b4d4c86041b554b85290701000000010000000000000000000000000000000000000000000000000000000000000000ffffffff07044c86041b0136ffffffff0100f2052a01000000434104eaafc2314def4ca98ac970241bcab022b9c1e1f4ea423a20f134c876f2c01ec0f0dd5b2e86e7168cefe0d81113c3807420ce13ad1357231a2252247d97a46a91ac000000000100000001bcad20a6a29827d1424f08989255120bf7f3e9e3cdaaa6bb31b0737fe048724300000000494830450220356e834b046cadc0f8ebb5a8a017b02de59c86305403dad52cd77b55af062ea10221009253cd6c119d4729b77c978e1e2aa19f5ea6e0e52b3f16e32fa608cd5bab753901ffffffff02008d380c010000001976a9142b4b8072ecbba129b6453c63e129e643207249ca88ac0065cd1d000000001976a9141b8dd13b994bcfc787b32aeadf58ccb3615cbd5488ac000000000100000003fdacf9b3eb077412e7a968d2e4f11b9a9dee312d666187ed77ee7d26af16cb0b000000008c493046022100ea1608e70911ca0de5af51ba57ad23b9a51db8d28f82c53563c56a05c20f5a87022100a8bdc8b4a8acc8634c6b420410150775eb7f2474f5615f7fccd65af30f310fbf01410465fdf49e29b06b9a1582287b6279014f834edc317695d125ef623c1cc3aaece245bd69fcad7508666e9c74a49dc9056d5fc14338ef38118dc4afae5fe2c585caffffffff309e1913634ecb50f3c4f83e96e70b2df071b497b8973a3e75429df397b5af83000000004948304502202bdb79c596a9ffc24e96f4386199aba386e9bc7b6071516e2b51dda942b3a1ed022100c53a857e76b724fc14d45311eac5019650d415c3abb5428f3aae16d8e69bec2301ffffffff2089e33491695080c9edc18a428f7d834db5b6d372df13ce2b1b0e0cbcb1e6c10000000049483045022100d4ce67c5896ee251c810ac1ff9ceccd328b497c8f553ab6e08431e7d40bad6b5022033119c0c2b7d792d31f1187779c7bd95aefd93d90a715586d73801d9b47471c601ffffffff0100714460030000001976a914c7b55141d097ea5df7a0ed330cf794376e53ec8d88ac0000000001000000045bf0e214aa4069a3e792ecee1e1bf0c1d397cde8dd08138f4b72a00681743447000000008b48304502200c45de8c4f3e2c1821f2fc878cba97b1e6f8807d94930713aa1c86a67b9bf1e40221008581abfef2e30f957815fc89978423746b2086375ca8ecf359c85c2a5b7c88ad01410462bb73f76ca0994fcb8b4271e6fb7561f5c0f9ca0cf6485261c4a0dc894f4ab844c6cdfb97cd0b60ffb5018ffd6238f4d87270efb1d3ae37079b794a92d7ec95ffffffffd669f7d7958d40fc59d2253d88e0f248e29b599c80bbcec344a83dda5f9aa72c000000008a473044022078124c8beeaa825f9e0b30bff96e564dd859432f2d0cb3b72d3d5d93d38d7e930220691d233b6c0f995be5acb03d70a7f7a65b6bc9bdd426260f38a1346669507a3601410462bb73f76ca0994fcb8b4271e6fb7561f5c0f9ca0cf6485261c4a0dc894f4ab844c6cdfb97cd0b60ffb5018ffd6238f4d87270efb1d3ae37079b794a92d7ec95fffffffff878af0d93f5229a68166cf051fd372bb7a537232946e0a46f53636b4dafdaa4000000008c493046022100c717d1714551663f69c3c5759bdbb3a0fcd3fab023abc0e522fe6440de35d8290221008d9cbe25bffc44af2b18e81c58eb37293fd7fe1c2e7b46fc37ee8c96c50ab1e201410462bb73f76ca0994fcb8b4271e6fb7561f5c0f9ca0cf6485261c4a0dc894f4ab844c6cdfb97cd0b60ffb5018ffd6238f4d87270efb1d3ae37079b794a92d7ec95ffffffff27f2b668859cd7f2f894aa0fd2d9e60963bcd07c88973f425f999b8cbfd7a1e2000000008c493046022100e00847147cbf517bcc2f502f3ddc6d284358d102ed20d47a8aa788a62f0db780022100d17b2d6fa84dcaf1c95d88d7e7c30385aecf415588d749afd3ec81f6022cecd701410462bb73f76ca0994fcb8b4271e6fb7561f5c0f9ca0cf6485261c4a0dc894f4ab844c6cdfb97cd0b60ffb5018ffd6238f4d87270efb1d3ae37079b794a92d7ec95ffffffff0100c817a8040000001976a914b6efd80d99179f4f4ff6f4dd0a007d018c385d2188ac000000000100000001834537b2f1ce8ef9373a258e10545ce5a50b758df616cd4356e0032554ebd3c4000000008b483045022100e68f422dd7c34fdce11eeb4509ddae38201773dd62f284e8aa9d96f85099d0b002202243bd399ff96b649a0fad05fa759d6a882f0af8c90cf7632c2840c29070aec20141045e58067e815c2f464c6a2a15f987758374203895710c2d452442e28496ff38ba8f5fd901dc20e29e88477167fe4fc299bf818fd0d9e1632d467b2a3d9503b1aaffffffff0280d7e636030000001976a914f34c3e10eb387efe872acb614c89e78bfca7815d88ac404b4c00000000001976a914a84e272933aaf87e1715d7786c51dfaeb5b65a6f88ac00000000010000000143ac81c8e6f6ef307dfe17f3d906d999e23e0189fda838c5510d850927e03ae7000000008c4930460221009c87c344760a64cb8ae6685a3eec2c1ac1bed5b88c87de51acd0e124f266c16602210082d07c037359c3a257b5c63ebd90f5a5edf97b2ac1c434b08ca998839f346dd40141040ba7e521fa7946d12edbb1d1e95a15c34bd4398195e86433c92b431cd315f455fe30032ede69cad9d1e1ed6c3c4ec0dbfced53438c625462afb792dcb098544bffffffff0240420f00000000001976a9144676d1b820d63ec272f1900d59d43bc6463d96f888ac40420f00000000001976a914648d04341d00d7968b3405c034adc38d4d8fb9bd88ac00000000010000000248cc917501ea5c55f4a8d2009c0567c40cfe037c2e71af017d0a452ff705e3f1000000008b483045022100bf5fdc86dc5f08a5d5c8e43a8c9d5b1ed8c65562e280007b52b133021acd9acc02205e325d613e555f772802bf413d36ba807892ed1a690a77811d3033b3de226e0a01410429fa713b124484cb2bd7b5557b2c0b9df7b2b1fee61825eadc5ae6c37a9920d38bfccdc7dc3cb0c47d7b173dbc9db8d37db0a33ae487982c59c6f8606e9d1791ffffffff41ed70551dd7e841883ab8f0b16bf04176b7d1480e4f0af9f3d4c3595768d068000000008b4830450221008513ad65187b903aed1102d1d0c47688127658c51106753fed0151ce9c16b80902201432b9ebcb87bd04ceb2de66035fbbaf4bf8b00d1cfe41f1a1f7338f9ad79d210141049d4cf80125bf50be1709f718c07ad15d0fc612b7da1f5570dddc35f2a352f0f27c978b06820edca9ef982c35fda2d255afba340068c5035552368bc7200c1488ffffffff0100093d00000000001976a9148edb68822f1ad580b043c7b3df2e400f8699eb4888ac00000000"), SER_NETWORK, PROTOCOL_VERSION);
     530           1 :     stream >> block;
     531             : 
     532           1 :     CBloomFilter filter(10, 0.000001, 0, BLOOM_UPDATE_NONE);
     533             :     // Match the generation pubkey
     534           1 :     filter.insert(ParseHex("04eaafc2314def4ca98ac970241bcab022b9c1e1f4ea423a20f134c876f2c01ec0f0dd5b2e86e7168cefe0d81113c3807420ce13ad1357231a2252247d97a46a91"));
     535             :     // ...and the output address of the 4th transaction
     536           1 :     filter.insert(ParseHex("b6efd80d99179f4f4ff6f4dd0a007d018c385d21"));
     537             : 
     538           1 :     CMerkleBlock merkleBlock(block, filter);
     539           1 :     BOOST_CHECK(merkleBlock.header.GetHash() == block.GetHash());
     540             : 
     541             :     // We shouldn't match any outpoints (UPDATE_NONE)
     542           1 :     BOOST_CHECK(!filter.contains(COutPoint(uint256S("0x147caa76786596590baa4e98f5d9f48b86c7765e489f7a6ff3360fe5c674360b"), 0)));
     543           1 :     BOOST_CHECK(!filter.contains(COutPoint(uint256S("0x02981fa052f0481dbc5868f4fc2166035a10f27a03cfd2de67326471df5bc041"), 0)));
     544           1 : }
     545             : 
     546       11398 : static std::vector<unsigned char> RandomData()
     547             : {
     548       11398 :     uint256 r = InsecureRand256();
     549       11398 :     return std::vector<unsigned char>(r.begin(), r.end());
     550             : }
     551             : 
     552         149 : BOOST_AUTO_TEST_CASE(rolling_bloom)
     553             : {
     554           1 :     SeedInsecureRand(SeedRand::ZEROS);
     555           1 :     g_mock_deterministic_tests = true;
     556             : 
     557             :     // last-100-entry, 1% false positive:
     558           1 :     CRollingBloomFilter rb1(100, 0.01);
     559             : 
     560             :     // Overfill:
     561             :     static const int DATASIZE=399;
     562           1 :     std::vector<unsigned char> data[DATASIZE];
     563         400 :     for (int i = 0; i < DATASIZE; i++) {
     564         399 :         data[i] = RandomData();
     565         399 :         rb1.insert(data[i]);
     566         399 :     }
     567             :     // Last 100 guaranteed to be remembered:
     568         101 :     for (int i = 299; i < DATASIZE; i++) {
     569         100 :         BOOST_CHECK(rb1.contains(data[i]));
     570         100 :     }
     571             : 
     572             :     // false positive rate is 1%, so we should get about 100 hits if
     573             :     // testing 10,000 random keys. We get worst-case false positive
     574             :     // behavior when the filter is as full as possible, which is
     575             :     // when we've inserted one minus an integer multiple of nElement*2.
     576           1 :     unsigned int nHits = 0;
     577       10001 :     for (int i = 0; i < 10000; i++) {
     578       10000 :         if (rb1.contains(RandomData()))
     579          75 :             ++nHits;
     580       10000 :     }
     581             :     // Expect about 100 hits
     582           1 :     BOOST_CHECK_EQUAL(nHits, 75U);
     583             : 
     584           1 :     BOOST_CHECK(rb1.contains(data[DATASIZE-1]));
     585           1 :     rb1.reset();
     586           1 :     BOOST_CHECK(!rb1.contains(data[DATASIZE-1]));
     587             : 
     588             :     // Now roll through data, make sure last 100 entries
     589             :     // are always remembered:
     590         400 :     for (int i = 0; i < DATASIZE; i++) {
     591         399 :         if (i >= 100)
     592         299 :             BOOST_CHECK(rb1.contains(data[i-100]));
     593         399 :         rb1.insert(data[i]);
     594         399 :         BOOST_CHECK(rb1.contains(data[i]));
     595         399 :     }
     596             : 
     597             :     // Insert 999 more random entries:
     598        1000 :     for (int i = 0; i < 999; i++) {
     599         999 :         std::vector<unsigned char> d = RandomData();
     600         999 :         rb1.insert(d);
     601         999 :         BOOST_CHECK(rb1.contains(d));
     602         999 :     }
     603             :     // Sanity check to make sure the filter isn't just filling up:
     604           1 :     nHits = 0;
     605         400 :     for (int i = 0; i < DATASIZE; i++) {
     606         399 :         if (rb1.contains(data[i]))
     607           6 :             ++nHits;
     608         399 :     }
     609             :     // Expect about 5 false positives
     610           1 :     BOOST_CHECK_EQUAL(nHits, 6U);
     611             : 
     612             :     // last-1000-entry, 0.01% false positive:
     613           1 :     CRollingBloomFilter rb2(1000, 0.001);
     614         400 :     for (int i = 0; i < DATASIZE; i++) {
     615         399 :         rb2.insert(data[i]);
     616         399 :     }
     617             :     // ... room for all of them:
     618         400 :     for (int i = 0; i < DATASIZE; i++) {
     619         399 :         BOOST_CHECK(rb2.contains(data[i]));
     620         399 :     }
     621           1 :     g_mock_deterministic_tests = false;
     622           1 : }
     623             : 
     624         146 : BOOST_AUTO_TEST_SUITE_END()

Generated by: LCOV version 1.16