|
|
@ -3,6 +3,8 @@ |
|
|
|
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
|
|
|
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
|
|
|
|
|
|
|
|
|
|
|
#include "key.h" |
|
|
|
#include "key.h" |
|
|
|
|
|
|
|
#include "keystore.h" |
|
|
|
|
|
|
|
#include "script/ismine.h" |
|
|
|
#include "script/script.h" |
|
|
|
#include "script/script.h" |
|
|
|
#include "script/script_error.h" |
|
|
|
#include "script/script_error.h" |
|
|
|
#include "script/standard.h" |
|
|
|
#include "script/standard.h" |
|
|
@ -81,7 +83,6 @@ BOOST_AUTO_TEST_CASE(script_standard_Solver_success) |
|
|
|
BOOST_CHECK(solutions[4] == std::vector<unsigned char>({3})); |
|
|
|
BOOST_CHECK(solutions[4] == std::vector<unsigned char>({3})); |
|
|
|
|
|
|
|
|
|
|
|
// TX_NULL_DATA
|
|
|
|
// TX_NULL_DATA
|
|
|
|
solutions.clear(); |
|
|
|
|
|
|
|
s.clear(); |
|
|
|
s.clear(); |
|
|
|
s << OP_RETURN << |
|
|
|
s << OP_RETURN << |
|
|
|
std::vector<unsigned char>({0}) << |
|
|
|
std::vector<unsigned char>({0}) << |
|
|
@ -92,7 +93,6 @@ BOOST_AUTO_TEST_CASE(script_standard_Solver_success) |
|
|
|
BOOST_CHECK_EQUAL(solutions.size(), 0); |
|
|
|
BOOST_CHECK_EQUAL(solutions.size(), 0); |
|
|
|
|
|
|
|
|
|
|
|
// TX_WITNESS_V0_KEYHASH
|
|
|
|
// TX_WITNESS_V0_KEYHASH
|
|
|
|
solutions.clear(); |
|
|
|
|
|
|
|
s.clear(); |
|
|
|
s.clear(); |
|
|
|
s << OP_0 << ToByteVector(pubkeys[0].GetID()); |
|
|
|
s << OP_0 << ToByteVector(pubkeys[0].GetID()); |
|
|
|
BOOST_CHECK(Solver(s, whichType, solutions)); |
|
|
|
BOOST_CHECK(Solver(s, whichType, solutions)); |
|
|
@ -102,9 +102,9 @@ BOOST_AUTO_TEST_CASE(script_standard_Solver_success) |
|
|
|
|
|
|
|
|
|
|
|
// TX_WITNESS_V0_SCRIPTHASH
|
|
|
|
// TX_WITNESS_V0_SCRIPTHASH
|
|
|
|
uint256 scriptHash; |
|
|
|
uint256 scriptHash; |
|
|
|
CSHA256().Write(&redeemScript[0], redeemScript.size()).Finalize(scriptHash.begin()); |
|
|
|
CSHA256().Write(&redeemScript[0], redeemScript.size()) |
|
|
|
|
|
|
|
.Finalize(scriptHash.begin()); |
|
|
|
|
|
|
|
|
|
|
|
solutions.clear(); |
|
|
|
|
|
|
|
s.clear(); |
|
|
|
s.clear(); |
|
|
|
s << OP_0 << ToByteVector(scriptHash); |
|
|
|
s << OP_0 << ToByteVector(scriptHash); |
|
|
|
BOOST_CHECK(Solver(s, whichType, solutions)); |
|
|
|
BOOST_CHECK(Solver(s, whichType, solutions)); |
|
|
@ -113,7 +113,6 @@ BOOST_AUTO_TEST_CASE(script_standard_Solver_success) |
|
|
|
BOOST_CHECK(solutions[0] == ToByteVector(scriptHash)); |
|
|
|
BOOST_CHECK(solutions[0] == ToByteVector(scriptHash)); |
|
|
|
|
|
|
|
|
|
|
|
// TX_NONSTANDARD
|
|
|
|
// TX_NONSTANDARD
|
|
|
|
solutions.clear(); |
|
|
|
|
|
|
|
s.clear(); |
|
|
|
s.clear(); |
|
|
|
s << OP_9 << OP_ADD << OP_11 << OP_EQUAL; |
|
|
|
s << OP_9 << OP_ADD << OP_11 << OP_EQUAL; |
|
|
|
BOOST_CHECK(!Solver(s, whichType, solutions)); |
|
|
|
BOOST_CHECK(!Solver(s, whichType, solutions)); |
|
|
@ -374,7 +373,8 @@ BOOST_AUTO_TEST_CASE(script_standard_GetScriptFor_) |
|
|
|
witnessScript << OP_1 << ToByteVector(pubkeys[0]) << OP_1 << OP_CHECKMULTISIG; |
|
|
|
witnessScript << OP_1 << ToByteVector(pubkeys[0]) << OP_1 << OP_CHECKMULTISIG; |
|
|
|
|
|
|
|
|
|
|
|
uint256 scriptHash; |
|
|
|
uint256 scriptHash; |
|
|
|
CSHA256().Write(&witnessScript[0], witnessScript.size()).Finalize(scriptHash.begin()); |
|
|
|
CSHA256().Write(&witnessScript[0], witnessScript.size()) |
|
|
|
|
|
|
|
.Finalize(scriptHash.begin()); |
|
|
|
|
|
|
|
|
|
|
|
expected.clear(); |
|
|
|
expected.clear(); |
|
|
|
expected << OP_0 << ToByteVector(scriptHash); |
|
|
|
expected << OP_0 << ToByteVector(scriptHash); |
|
|
@ -382,4 +382,359 @@ BOOST_AUTO_TEST_CASE(script_standard_GetScriptFor_) |
|
|
|
BOOST_CHECK(result == expected); |
|
|
|
BOOST_CHECK(result == expected); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(script_standard_IsMine) |
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
CKey keys[2]; |
|
|
|
|
|
|
|
CPubKey pubkeys[2]; |
|
|
|
|
|
|
|
for (int i = 0; i < 2; i++) { |
|
|
|
|
|
|
|
keys[i].MakeNewKey(true); |
|
|
|
|
|
|
|
pubkeys[i] = keys[i].GetPubKey(); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
CKey uncompressedKey; |
|
|
|
|
|
|
|
uncompressedKey.MakeNewKey(false); |
|
|
|
|
|
|
|
CPubKey uncompressedPubkey = uncompressedKey.GetPubKey(); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
CScript scriptPubKey; |
|
|
|
|
|
|
|
isminetype result; |
|
|
|
|
|
|
|
bool isInvalid; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// P2PK compressed
|
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
CBasicKeyStore keystore; |
|
|
|
|
|
|
|
scriptPubKey.clear(); |
|
|
|
|
|
|
|
scriptPubKey << ToByteVector(pubkeys[0]) << OP_CHECKSIG; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Keystore does not have key
|
|
|
|
|
|
|
|
result = IsMine(keystore, scriptPubKey, isInvalid); |
|
|
|
|
|
|
|
BOOST_CHECK_EQUAL(result, ISMINE_NO); |
|
|
|
|
|
|
|
BOOST_CHECK(!isInvalid); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Keystore has key
|
|
|
|
|
|
|
|
keystore.AddKey(keys[0]); |
|
|
|
|
|
|
|
result = IsMine(keystore, scriptPubKey, isInvalid); |
|
|
|
|
|
|
|
BOOST_CHECK_EQUAL(result, ISMINE_SPENDABLE); |
|
|
|
|
|
|
|
BOOST_CHECK(!isInvalid); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// P2PK uncompressed
|
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
CBasicKeyStore keystore; |
|
|
|
|
|
|
|
scriptPubKey.clear(); |
|
|
|
|
|
|
|
scriptPubKey << ToByteVector(uncompressedPubkey) << OP_CHECKSIG; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Keystore does not have key
|
|
|
|
|
|
|
|
result = IsMine(keystore, scriptPubKey, isInvalid); |
|
|
|
|
|
|
|
BOOST_CHECK_EQUAL(result, ISMINE_NO); |
|
|
|
|
|
|
|
BOOST_CHECK(!isInvalid); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Keystore has key
|
|
|
|
|
|
|
|
keystore.AddKey(uncompressedKey); |
|
|
|
|
|
|
|
result = IsMine(keystore, scriptPubKey, isInvalid); |
|
|
|
|
|
|
|
BOOST_CHECK_EQUAL(result, ISMINE_SPENDABLE); |
|
|
|
|
|
|
|
BOOST_CHECK(!isInvalid); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// P2PKH compressed
|
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
CBasicKeyStore keystore; |
|
|
|
|
|
|
|
scriptPubKey.clear(); |
|
|
|
|
|
|
|
scriptPubKey << OP_DUP << OP_HASH160 << ToByteVector(pubkeys[0].GetID()) << OP_EQUALVERIFY << OP_CHECKSIG; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Keystore does not have key
|
|
|
|
|
|
|
|
result = IsMine(keystore, scriptPubKey, isInvalid); |
|
|
|
|
|
|
|
BOOST_CHECK_EQUAL(result, ISMINE_NO); |
|
|
|
|
|
|
|
BOOST_CHECK(!isInvalid); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Keystore has key
|
|
|
|
|
|
|
|
keystore.AddKey(keys[0]); |
|
|
|
|
|
|
|
result = IsMine(keystore, scriptPubKey, isInvalid); |
|
|
|
|
|
|
|
BOOST_CHECK_EQUAL(result, ISMINE_SPENDABLE); |
|
|
|
|
|
|
|
BOOST_CHECK(!isInvalid); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// P2PKH uncompressed
|
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
CBasicKeyStore keystore; |
|
|
|
|
|
|
|
scriptPubKey.clear(); |
|
|
|
|
|
|
|
scriptPubKey << OP_DUP << OP_HASH160 << ToByteVector(uncompressedPubkey.GetID()) << OP_EQUALVERIFY << OP_CHECKSIG; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Keystore does not have key
|
|
|
|
|
|
|
|
result = IsMine(keystore, scriptPubKey, isInvalid); |
|
|
|
|
|
|
|
BOOST_CHECK_EQUAL(result, ISMINE_NO); |
|
|
|
|
|
|
|
BOOST_CHECK(!isInvalid); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Keystore has key
|
|
|
|
|
|
|
|
keystore.AddKey(uncompressedKey); |
|
|
|
|
|
|
|
result = IsMine(keystore, scriptPubKey, isInvalid); |
|
|
|
|
|
|
|
BOOST_CHECK_EQUAL(result, ISMINE_SPENDABLE); |
|
|
|
|
|
|
|
BOOST_CHECK(!isInvalid); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// P2SH
|
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
CBasicKeyStore keystore; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
CScript redeemScript; |
|
|
|
|
|
|
|
redeemScript << OP_DUP << OP_HASH160 << ToByteVector(pubkeys[0].GetID()) << OP_EQUALVERIFY << OP_CHECKSIG; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
scriptPubKey.clear(); |
|
|
|
|
|
|
|
scriptPubKey << OP_HASH160 << ToByteVector(CScriptID(redeemScript)) << OP_EQUAL; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Keystore does not have redeemScript or key
|
|
|
|
|
|
|
|
result = IsMine(keystore, scriptPubKey, isInvalid); |
|
|
|
|
|
|
|
BOOST_CHECK_EQUAL(result, ISMINE_NO); |
|
|
|
|
|
|
|
BOOST_CHECK(!isInvalid); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Keystore has redeemScript but no key
|
|
|
|
|
|
|
|
keystore.AddCScript(redeemScript); |
|
|
|
|
|
|
|
result = IsMine(keystore, scriptPubKey, isInvalid); |
|
|
|
|
|
|
|
BOOST_CHECK_EQUAL(result, ISMINE_NO); |
|
|
|
|
|
|
|
BOOST_CHECK(!isInvalid); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Keystore has redeemScript and key
|
|
|
|
|
|
|
|
keystore.AddKey(keys[0]); |
|
|
|
|
|
|
|
result = IsMine(keystore, scriptPubKey, isInvalid); |
|
|
|
|
|
|
|
BOOST_CHECK_EQUAL(result, ISMINE_SPENDABLE); |
|
|
|
|
|
|
|
BOOST_CHECK(!isInvalid); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// P2WPKH compressed
|
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
CBasicKeyStore keystore; |
|
|
|
|
|
|
|
keystore.AddKey(keys[0]); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
scriptPubKey.clear(); |
|
|
|
|
|
|
|
scriptPubKey << OP_0 << ToByteVector(pubkeys[0].GetID()); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Keystore has key, but no P2SH redeemScript
|
|
|
|
|
|
|
|
result = IsMine(keystore, scriptPubKey, isInvalid); |
|
|
|
|
|
|
|
BOOST_CHECK_EQUAL(result, ISMINE_NO); |
|
|
|
|
|
|
|
BOOST_CHECK(!isInvalid); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Keystore has key and P2SH redeemScript
|
|
|
|
|
|
|
|
keystore.AddCScript(scriptPubKey); |
|
|
|
|
|
|
|
result = IsMine(keystore, scriptPubKey, isInvalid); |
|
|
|
|
|
|
|
BOOST_CHECK_EQUAL(result, ISMINE_SPENDABLE); |
|
|
|
|
|
|
|
BOOST_CHECK(!isInvalid); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// P2WPKH uncompressed
|
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
CBasicKeyStore keystore; |
|
|
|
|
|
|
|
keystore.AddKey(uncompressedKey); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
scriptPubKey.clear(); |
|
|
|
|
|
|
|
scriptPubKey << OP_0 << ToByteVector(uncompressedPubkey.GetID()); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Keystore has key, but no P2SH redeemScript
|
|
|
|
|
|
|
|
result = IsMine(keystore, scriptPubKey, isInvalid); |
|
|
|
|
|
|
|
BOOST_CHECK_EQUAL(result, ISMINE_NO); |
|
|
|
|
|
|
|
BOOST_CHECK(!isInvalid); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Keystore has key and P2SH redeemScript
|
|
|
|
|
|
|
|
keystore.AddCScript(scriptPubKey); |
|
|
|
|
|
|
|
result = IsMine(keystore, scriptPubKey, isInvalid); |
|
|
|
|
|
|
|
BOOST_CHECK_EQUAL(result, ISMINE_NO); |
|
|
|
|
|
|
|
BOOST_CHECK(isInvalid); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// scriptPubKey multisig
|
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
CBasicKeyStore keystore; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
scriptPubKey.clear(); |
|
|
|
|
|
|
|
scriptPubKey << OP_2 << |
|
|
|
|
|
|
|
ToByteVector(uncompressedPubkey) << |
|
|
|
|
|
|
|
ToByteVector(pubkeys[1]) << |
|
|
|
|
|
|
|
OP_2 << OP_CHECKMULTISIG; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Keystore does not have any keys
|
|
|
|
|
|
|
|
result = IsMine(keystore, scriptPubKey, isInvalid); |
|
|
|
|
|
|
|
BOOST_CHECK_EQUAL(result, ISMINE_NO); |
|
|
|
|
|
|
|
BOOST_CHECK(!isInvalid); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Keystore has 1/2 keys
|
|
|
|
|
|
|
|
keystore.AddKey(uncompressedKey); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
result = IsMine(keystore, scriptPubKey, isInvalid); |
|
|
|
|
|
|
|
BOOST_CHECK_EQUAL(result, ISMINE_NO); |
|
|
|
|
|
|
|
BOOST_CHECK(!isInvalid); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Keystore has 2/2 keys
|
|
|
|
|
|
|
|
keystore.AddKey(keys[1]); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
result = IsMine(keystore, scriptPubKey, isInvalid); |
|
|
|
|
|
|
|
BOOST_CHECK_EQUAL(result, ISMINE_SPENDABLE); |
|
|
|
|
|
|
|
BOOST_CHECK(!isInvalid); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// P2SH multisig
|
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
CBasicKeyStore keystore; |
|
|
|
|
|
|
|
keystore.AddKey(uncompressedKey); |
|
|
|
|
|
|
|
keystore.AddKey(keys[1]); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
CScript redeemScript; |
|
|
|
|
|
|
|
redeemScript << OP_2 << |
|
|
|
|
|
|
|
ToByteVector(uncompressedPubkey) << |
|
|
|
|
|
|
|
ToByteVector(pubkeys[1]) << |
|
|
|
|
|
|
|
OP_2 << OP_CHECKMULTISIG; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
scriptPubKey.clear(); |
|
|
|
|
|
|
|
scriptPubKey << OP_HASH160 << ToByteVector(CScriptID(redeemScript)) << OP_EQUAL; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Keystore has no redeemScript
|
|
|
|
|
|
|
|
result = IsMine(keystore, scriptPubKey, isInvalid); |
|
|
|
|
|
|
|
BOOST_CHECK_EQUAL(result, ISMINE_NO); |
|
|
|
|
|
|
|
BOOST_CHECK(!isInvalid); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Keystore has redeemScript
|
|
|
|
|
|
|
|
keystore.AddCScript(redeemScript); |
|
|
|
|
|
|
|
result = IsMine(keystore, scriptPubKey, isInvalid); |
|
|
|
|
|
|
|
BOOST_CHECK_EQUAL(result, ISMINE_SPENDABLE); |
|
|
|
|
|
|
|
BOOST_CHECK(!isInvalid); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// P2WSH multisig with compressed keys
|
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
CBasicKeyStore keystore; |
|
|
|
|
|
|
|
keystore.AddKey(keys[0]); |
|
|
|
|
|
|
|
keystore.AddKey(keys[1]); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
CScript witnessScript; |
|
|
|
|
|
|
|
witnessScript << OP_2 << |
|
|
|
|
|
|
|
ToByteVector(pubkeys[0]) << |
|
|
|
|
|
|
|
ToByteVector(pubkeys[1]) << |
|
|
|
|
|
|
|
OP_2 << OP_CHECKMULTISIG; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
uint256 scriptHash; |
|
|
|
|
|
|
|
CSHA256().Write(&witnessScript[0], witnessScript.size()) |
|
|
|
|
|
|
|
.Finalize(scriptHash.begin()); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
scriptPubKey.clear(); |
|
|
|
|
|
|
|
scriptPubKey << OP_0 << ToByteVector(scriptHash); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Keystore has keys, but no witnessScript or P2SH redeemScript
|
|
|
|
|
|
|
|
result = IsMine(keystore, scriptPubKey, isInvalid); |
|
|
|
|
|
|
|
BOOST_CHECK_EQUAL(result, ISMINE_NO); |
|
|
|
|
|
|
|
BOOST_CHECK(!isInvalid); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Keystore has keys and witnessScript, but no P2SH redeemScript
|
|
|
|
|
|
|
|
keystore.AddCScript(witnessScript); |
|
|
|
|
|
|
|
result = IsMine(keystore, scriptPubKey, isInvalid); |
|
|
|
|
|
|
|
BOOST_CHECK_EQUAL(result, ISMINE_NO); |
|
|
|
|
|
|
|
BOOST_CHECK(!isInvalid); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Keystore has keys, witnessScript, P2SH redeemScript
|
|
|
|
|
|
|
|
keystore.AddCScript(scriptPubKey); |
|
|
|
|
|
|
|
result = IsMine(keystore, scriptPubKey, isInvalid); |
|
|
|
|
|
|
|
BOOST_CHECK_EQUAL(result, ISMINE_SPENDABLE); |
|
|
|
|
|
|
|
BOOST_CHECK(!isInvalid); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// P2WSH multisig with uncompressed key
|
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
CBasicKeyStore keystore; |
|
|
|
|
|
|
|
keystore.AddKey(uncompressedKey); |
|
|
|
|
|
|
|
keystore.AddKey(keys[1]); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
CScript witnessScript; |
|
|
|
|
|
|
|
witnessScript << OP_2 << |
|
|
|
|
|
|
|
ToByteVector(uncompressedPubkey) << |
|
|
|
|
|
|
|
ToByteVector(pubkeys[1]) << |
|
|
|
|
|
|
|
OP_2 << OP_CHECKMULTISIG; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
uint256 scriptHash; |
|
|
|
|
|
|
|
CSHA256().Write(&witnessScript[0], witnessScript.size()) |
|
|
|
|
|
|
|
.Finalize(scriptHash.begin()); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
scriptPubKey.clear(); |
|
|
|
|
|
|
|
scriptPubKey << OP_0 << ToByteVector(scriptHash); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Keystore has keys, but no witnessScript or P2SH redeemScript
|
|
|
|
|
|
|
|
result = IsMine(keystore, scriptPubKey, isInvalid); |
|
|
|
|
|
|
|
BOOST_CHECK_EQUAL(result, ISMINE_NO); |
|
|
|
|
|
|
|
BOOST_CHECK(!isInvalid); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Keystore has keys and witnessScript, but no P2SH redeemScript
|
|
|
|
|
|
|
|
keystore.AddCScript(witnessScript); |
|
|
|
|
|
|
|
result = IsMine(keystore, scriptPubKey, isInvalid); |
|
|
|
|
|
|
|
BOOST_CHECK_EQUAL(result, ISMINE_NO); |
|
|
|
|
|
|
|
BOOST_CHECK(!isInvalid); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Keystore has keys, witnessScript, P2SH redeemScript
|
|
|
|
|
|
|
|
keystore.AddCScript(scriptPubKey); |
|
|
|
|
|
|
|
result = IsMine(keystore, scriptPubKey, isInvalid); |
|
|
|
|
|
|
|
BOOST_CHECK_EQUAL(result, ISMINE_NO); |
|
|
|
|
|
|
|
BOOST_CHECK(isInvalid); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// P2WSH multisig wrapped in P2SH
|
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
CBasicKeyStore keystore; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
CScript witnessScript; |
|
|
|
|
|
|
|
witnessScript << OP_2 << |
|
|
|
|
|
|
|
ToByteVector(pubkeys[0]) << |
|
|
|
|
|
|
|
ToByteVector(pubkeys[1]) << |
|
|
|
|
|
|
|
OP_2 << OP_CHECKMULTISIG; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
uint256 scriptHash; |
|
|
|
|
|
|
|
CSHA256().Write(&witnessScript[0], witnessScript.size()) |
|
|
|
|
|
|
|
.Finalize(scriptHash.begin()); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
CScript redeemScript; |
|
|
|
|
|
|
|
redeemScript << OP_0 << ToByteVector(scriptHash); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
scriptPubKey.clear(); |
|
|
|
|
|
|
|
scriptPubKey << OP_HASH160 << ToByteVector(CScriptID(redeemScript)) << OP_EQUAL; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Keystore has no witnessScript, P2SH redeemScript, or keys
|
|
|
|
|
|
|
|
result = IsMine(keystore, scriptPubKey, isInvalid); |
|
|
|
|
|
|
|
BOOST_CHECK_EQUAL(result, ISMINE_NO); |
|
|
|
|
|
|
|
BOOST_CHECK(!isInvalid); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Keystore has witnessScript and P2SH redeemScript, but no keys
|
|
|
|
|
|
|
|
keystore.AddCScript(redeemScript); |
|
|
|
|
|
|
|
keystore.AddCScript(witnessScript); |
|
|
|
|
|
|
|
result = IsMine(keystore, scriptPubKey, isInvalid); |
|
|
|
|
|
|
|
BOOST_CHECK_EQUAL(result, ISMINE_NO); |
|
|
|
|
|
|
|
BOOST_CHECK(!isInvalid); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Keystore has keys, witnessScript, P2SH redeemScript
|
|
|
|
|
|
|
|
keystore.AddKey(keys[0]); |
|
|
|
|
|
|
|
keystore.AddKey(keys[1]); |
|
|
|
|
|
|
|
result = IsMine(keystore, scriptPubKey, isInvalid); |
|
|
|
|
|
|
|
BOOST_CHECK_EQUAL(result, ISMINE_SPENDABLE); |
|
|
|
|
|
|
|
BOOST_CHECK(!isInvalid); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// OP_RETURN
|
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
CBasicKeyStore keystore; |
|
|
|
|
|
|
|
keystore.AddKey(keys[0]); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
scriptPubKey.clear(); |
|
|
|
|
|
|
|
scriptPubKey << OP_RETURN << ToByteVector(pubkeys[0]); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
result = IsMine(keystore, scriptPubKey, isInvalid); |
|
|
|
|
|
|
|
BOOST_CHECK_EQUAL(result, ISMINE_NO); |
|
|
|
|
|
|
|
BOOST_CHECK(!isInvalid); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Nonstandard
|
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
CBasicKeyStore keystore; |
|
|
|
|
|
|
|
keystore.AddKey(keys[0]); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
scriptPubKey.clear(); |
|
|
|
|
|
|
|
scriptPubKey << OP_9 << OP_ADD << OP_11 << OP_EQUAL; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
result = IsMine(keystore, scriptPubKey, isInvalid); |
|
|
|
|
|
|
|
BOOST_CHECK_EQUAL(result, ISMINE_NO); |
|
|
|
|
|
|
|
BOOST_CHECK(!isInvalid); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_SUITE_END() |
|
|
|
BOOST_AUTO_TEST_SUITE_END() |
|
|
|