2014-10-09 01:29:45 +02:00
// Copyright (c) 2011-2014 The Bitcoin Core developers
// Distributed under the MIT software license, see the accompanying
2014-03-18 10:11:00 +01:00
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
2013-04-13 07:13:08 +02:00
# include "data/script_invalid.json.h"
# include "data/script_valid.json.h"
2014-09-20 03:13:04 +02:00
# include "core_io.h"
2013-04-13 07:13:08 +02:00
# include "key.h"
# include "keystore.h"
# include "main.h"
2014-08-20 17:37:40 +02:00
# include "script/script.h"
2014-11-13 20:27:38 +01:00
# include "script/script_error.h"
2014-08-27 17:22:33 +02:00
# include "script/sign.h"
2014-09-20 03:13:04 +02:00
# include "util.h"
2015-03-12 09:34:42 +01:00
# include "test/test_bitcoin.h"
2013-04-13 07:13:08 +02:00
2014-10-15 00:23:46 +02:00
# if defined(HAVE_CONSENSUS_LIB)
# include "script/bitcoinconsensus.h"
# endif
2012-04-17 23:57:06 +02:00
# include <fstream>
2013-04-13 07:13:08 +02:00
# include <stdint.h>
# include <string>
2011-07-31 20:01:31 +02:00
# include <vector>
2013-04-13 07:13:08 +02:00
2011-07-31 20:01:31 +02:00
# include <boost/foreach.hpp>
2012-04-26 17:20:44 +02:00
# include <boost/test/unit_test.hpp>
2014-08-20 21:15:16 +02:00
# include "json_spirit_wrapper.h"
2011-07-31 20:01:31 +02:00
using namespace std ;
2012-04-17 23:57:06 +02:00
using namespace json_spirit ;
2014-10-07 02:22:47 +02:00
// Uncomment if you want to output updated JSON tests.
// #define UPDATE_JSON_TESTS
2012-11-13 23:03:25 +01:00
static const unsigned int flags = SCRIPT_VERIFY_P2SH | SCRIPT_VERIFY_STRICTENC ;
2011-07-31 20:01:31 +02:00
2014-09-08 21:15:31 +02:00
unsigned int ParseScriptFlags ( string strFlags ) ;
2014-09-20 03:13:04 +02:00
string FormatScriptFlags ( unsigned int flags ) ;
2014-09-08 21:15:31 +02:00
2012-04-17 23:57:06 +02:00
Array
2013-09-10 21:18:09 +02:00
read_json ( const std : : string & jsondata )
2012-04-17 23:57:06 +02:00
{
Value v ;
2013-09-10 21:18:09 +02:00
2014-08-20 21:15:16 +02:00
if ( ! v . read ( jsondata ) | | ! v . isArray ( ) )
2012-04-17 23:57:06 +02:00
{
2013-09-10 21:18:09 +02:00
BOOST_ERROR ( " Parse error. " ) ;
2012-04-17 23:57:06 +02:00
return Array ( ) ;
}
return v . get_array ( ) ;
}
2015-03-12 09:34:42 +01:00
BOOST_FIXTURE_TEST_SUITE ( script_tests , BasicTestingSetup )
2011-07-31 20:01:31 +02:00
2014-09-20 00:20:53 +02:00
CMutableTransaction BuildCreditingTransaction ( const CScript & scriptPubKey )
{
CMutableTransaction txCredit ;
txCredit . nVersion = 1 ;
txCredit . nLockTime = 0 ;
txCredit . vin . resize ( 1 ) ;
txCredit . vout . resize ( 1 ) ;
txCredit . vin [ 0 ] . prevout . SetNull ( ) ;
txCredit . vin [ 0 ] . scriptSig = CScript ( ) < < CScriptNum ( 0 ) < < CScriptNum ( 0 ) ;
txCredit . vin [ 0 ] . nSequence = std : : numeric_limits < unsigned int > : : max ( ) ;
txCredit . vout [ 0 ] . scriptPubKey = scriptPubKey ;
txCredit . vout [ 0 ] . nValue = 0 ;
return txCredit ;
}
2014-09-28 06:33:33 +02:00
CMutableTransaction BuildSpendingTransaction ( const CScript & scriptSig , const CMutableTransaction & txCredit )
2014-09-20 00:20:53 +02:00
{
CMutableTransaction txSpend ;
txSpend . nVersion = 1 ;
txSpend . nLockTime = 0 ;
txSpend . vin . resize ( 1 ) ;
txSpend . vout . resize ( 1 ) ;
txSpend . vin [ 0 ] . prevout . hash = txCredit . GetHash ( ) ;
txSpend . vin [ 0 ] . prevout . n = 0 ;
txSpend . vin [ 0 ] . scriptSig = scriptSig ;
txSpend . vin [ 0 ] . nSequence = std : : numeric_limits < unsigned int > : : max ( ) ;
txSpend . vout [ 0 ] . scriptPubKey = CScript ( ) ;
txSpend . vout [ 0 ] . nValue = 0 ;
return txSpend ;
}
2014-09-20 03:13:04 +02:00
void DoTest ( const CScript & scriptPubKey , const CScript & scriptSig , int flags , bool expect , const std : : string & message )
{
2014-11-13 20:27:38 +01:00
ScriptError err ;
2014-10-15 00:23:46 +02:00
CMutableTransaction tx = BuildSpendingTransaction ( scriptSig , BuildCreditingTransaction ( scriptPubKey ) ) ;
CMutableTransaction tx2 = tx ;
2015-01-27 15:01:31 +01:00
BOOST_CHECK_MESSAGE ( VerifyScript ( scriptSig , scriptPubKey , flags , MutableTransactionSignatureChecker ( & tx , 0 ) , & err ) = = expect , message ) ;
2014-11-13 20:27:38 +01:00
BOOST_CHECK_MESSAGE ( expect = = ( err = = SCRIPT_ERR_OK ) , std : : string ( ScriptErrorString ( err ) ) + " : " + message ) ;
2014-10-15 00:23:46 +02:00
# if defined(HAVE_CONSENSUS_LIB)
CDataStream stream ( SER_NETWORK , PROTOCOL_VERSION ) ;
stream < < tx2 ;
BOOST_CHECK_MESSAGE ( bitcoinconsensus_verify_script ( begin_ptr ( scriptPubKey ) , scriptPubKey . size ( ) , ( const unsigned char * ) & stream [ 0 ] , stream . size ( ) , 0 , flags , NULL ) = = expect , message ) ;
# endif
2014-09-20 03:13:04 +02:00
}
2014-11-05 19:53:59 +01:00
void static NegateSignatureS ( std : : vector < unsigned char > & vchSig ) {
// Parse the signature.
std : : vector < unsigned char > r , s ;
r = std : : vector < unsigned char > ( vchSig . begin ( ) + 4 , vchSig . begin ( ) + 4 + vchSig [ 3 ] ) ;
s = std : : vector < unsigned char > ( vchSig . begin ( ) + 6 + vchSig [ 3 ] , vchSig . begin ( ) + 6 + vchSig [ 3 ] + vchSig [ 5 + vchSig [ 3 ] ] ) ;
// Really ugly to implement mod-n negation here, but it would be feature creep to expose such functionality from libsecp256k1.
static const unsigned char order [ 33 ] = {
0x00 ,
0xFF , 0xFF , 0xFF , 0xFF , 0xFF , 0xFF , 0xFF , 0xFF ,
0xFF , 0xFF , 0xFF , 0xFF , 0xFF , 0xFF , 0xFF , 0xFE ,
0xBA , 0xAE , 0xDC , 0xE6 , 0xAF , 0x48 , 0xA0 , 0x3B ,
0xBF , 0xD2 , 0x5E , 0x8C , 0xD0 , 0x36 , 0x41 , 0x41
} ;
while ( s . size ( ) < 33 ) {
s . insert ( s . begin ( ) , 0x00 ) ;
}
int carry = 0 ;
for ( int p = 32 ; p > = 1 ; p - - ) {
int n = ( int ) order [ p ] - s [ p ] - carry ;
s [ p ] = ( n + 256 ) & 0xFF ;
carry = ( n < 0 ) ;
}
assert ( carry = = 0 ) ;
if ( s . size ( ) > 1 & & s [ 0 ] = = 0 & & s [ 1 ] < 0x80 ) {
s . erase ( s . begin ( ) ) ;
}
// Reconstruct the signature.
vchSig . clear ( ) ;
vchSig . push_back ( 0x30 ) ;
vchSig . push_back ( 4 + r . size ( ) + s . size ( ) ) ;
vchSig . push_back ( 0x02 ) ;
vchSig . push_back ( r . size ( ) ) ;
vchSig . insert ( vchSig . end ( ) , r . begin ( ) , r . end ( ) ) ;
vchSig . push_back ( 0x02 ) ;
vchSig . push_back ( s . size ( ) ) ;
vchSig . insert ( vchSig . end ( ) , s . begin ( ) , s . end ( ) ) ;
}
2014-09-20 03:13:04 +02:00
namespace
{
const unsigned char vchKey0 [ 32 ] = { 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 1 } ;
const unsigned char vchKey1 [ 32 ] = { 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 1 , 0 } ;
const unsigned char vchKey2 [ 32 ] = { 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 1 , 0 , 0 } ;
struct KeyData
{
CKey key0 , key0C , key1 , key1C , key2 , key2C ;
CPubKey pubkey0 , pubkey0C , pubkey0H ;
CPubKey pubkey1 , pubkey1C ;
CPubKey pubkey2 , pubkey2C ;
KeyData ( )
{
2014-09-29 19:16:24 +02:00
key0 . Set ( vchKey0 , vchKey0 + 32 , false ) ;
key0C . Set ( vchKey0 , vchKey0 + 32 , true ) ;
2014-09-20 03:13:04 +02:00
pubkey0 = key0 . GetPubKey ( ) ;
pubkey0H = key0 . GetPubKey ( ) ;
pubkey0C = key0C . GetPubKey ( ) ;
* const_cast < unsigned char * > ( & pubkey0H [ 0 ] ) = 0x06 | ( pubkey0H [ 64 ] & 1 ) ;
2014-09-29 19:16:24 +02:00
key1 . Set ( vchKey1 , vchKey1 + 32 , false ) ;
key1C . Set ( vchKey1 , vchKey1 + 32 , true ) ;
2014-09-20 03:13:04 +02:00
pubkey1 = key1 . GetPubKey ( ) ;
pubkey1C = key1C . GetPubKey ( ) ;
2014-09-29 19:16:24 +02:00
key2 . Set ( vchKey2 , vchKey2 + 32 , false ) ;
key2C . Set ( vchKey2 , vchKey2 + 32 , true ) ;
2014-09-20 03:13:04 +02:00
pubkey2 = key2 . GetPubKey ( ) ;
pubkey2C = key2C . GetPubKey ( ) ;
}
} ;
class TestBuilder
{
private :
CScript scriptPubKey ;
CTransaction creditTx ;
CMutableTransaction spendTx ;
bool havePush ;
std : : vector < unsigned char > push ;
std : : string comment ;
int flags ;
void DoPush ( )
{
if ( havePush ) {
spendTx . vin [ 0 ] . scriptSig < < push ;
havePush = false ;
}
}
void DoPush ( const std : : vector < unsigned char > & data )
{
DoPush ( ) ;
push = data ;
havePush = true ;
}
public :
TestBuilder ( const CScript & redeemScript , const std : : string & comment_ , int flags_ , bool P2SH = false ) : scriptPubKey ( redeemScript ) , havePush ( false ) , comment ( comment_ ) , flags ( flags_ )
{
if ( P2SH ) {
2014-09-25 04:54:08 +02:00
creditTx = BuildCreditingTransaction ( CScript ( ) < < OP_HASH160 < < ToByteVector ( CScriptID ( redeemScript ) ) < < OP_EQUAL ) ;
2014-09-20 03:13:04 +02:00
} else {
creditTx = BuildCreditingTransaction ( redeemScript ) ;
}
spendTx = BuildSpendingTransaction ( CScript ( ) , creditTx ) ;
}
TestBuilder & Add ( const CScript & script )
{
2014-10-09 01:29:45 +02:00
DoPush ( ) ;
2014-09-20 03:13:04 +02:00
spendTx . vin [ 0 ] . scriptSig + = script ;
return * this ;
}
TestBuilder & Num ( int num )
{
2014-10-09 01:29:45 +02:00
DoPush ( ) ;
2014-10-09 03:48:59 +02:00
spendTx . vin [ 0 ] . scriptSig < < num ;
2014-09-20 03:13:04 +02:00
return * this ;
}
TestBuilder & Push ( const std : : string & hex )
{
DoPush ( ParseHex ( hex ) ) ;
return * this ;
}
TestBuilder & PushSig ( const CKey & key , int nHashType = SIGHASH_ALL , unsigned int lenR = 32 , unsigned int lenS = 32 )
{
uint256 hash = SignatureHash ( scriptPubKey , spendTx , 0 , nHashType ) ;
std : : vector < unsigned char > vchSig , r , s ;
2014-11-06 15:54:50 +01:00
uint32_t iter = 0 ;
2014-09-20 03:13:04 +02:00
do {
2014-11-06 15:54:50 +01:00
key . Sign ( hash , vchSig , iter + + ) ;
2014-11-05 19:53:59 +01:00
if ( ( lenS = = 33 ) ! = ( vchSig [ 5 + vchSig [ 3 ] ] = = 33 ) ) {
NegateSignatureS ( vchSig ) ;
}
2014-09-29 19:16:24 +02:00
r = std : : vector < unsigned char > ( vchSig . begin ( ) + 4 , vchSig . begin ( ) + 4 + vchSig [ 3 ] ) ;
s = std : : vector < unsigned char > ( vchSig . begin ( ) + 6 + vchSig [ 3 ] , vchSig . begin ( ) + 6 + vchSig [ 3 ] + vchSig [ 5 + vchSig [ 3 ] ] ) ;
2014-09-20 03:13:04 +02:00
} while ( lenR ! = r . size ( ) | | lenS ! = s . size ( ) ) ;
vchSig . push_back ( static_cast < unsigned char > ( nHashType ) ) ;
DoPush ( vchSig ) ;
return * this ;
}
TestBuilder & Push ( const CPubKey & pubkey )
{
DoPush ( std : : vector < unsigned char > ( pubkey . begin ( ) , pubkey . end ( ) ) ) ;
return * this ;
}
TestBuilder & PushRedeem ( )
{
DoPush ( static_cast < std : : vector < unsigned char > > ( scriptPubKey ) ) ;
return * this ;
}
TestBuilder & EditPush ( unsigned int pos , const std : : string & hexin , const std : : string & hexout )
{
assert ( havePush ) ;
std : : vector < unsigned char > datain = ParseHex ( hexin ) ;
std : : vector < unsigned char > dataout = ParseHex ( hexout ) ;
assert ( pos + datain . size ( ) < = push . size ( ) ) ;
BOOST_CHECK_MESSAGE ( std : : vector < unsigned char > ( push . begin ( ) + pos , push . begin ( ) + pos + datain . size ( ) ) = = datain , comment ) ;
push . erase ( push . begin ( ) + pos , push . begin ( ) + pos + datain . size ( ) ) ;
push . insert ( push . begin ( ) + pos , dataout . begin ( ) , dataout . end ( ) ) ;
return * this ;
}
TestBuilder & DamagePush ( unsigned int pos )
{
assert ( havePush ) ;
assert ( pos < push . size ( ) ) ;
push [ pos ] ^ = 1 ;
return * this ;
}
TestBuilder & Test ( bool expect )
{
TestBuilder copy = * this ; // Make a copy so we can rollback the push.
DoPush ( ) ;
DoTest ( creditTx . vout [ 0 ] . scriptPubKey , spendTx . vin [ 0 ] . scriptSig , flags , expect , comment ) ;
* this = copy ;
return * this ;
}
2014-10-07 02:22:47 +02:00
Array GetJSON ( )
2014-09-20 03:13:04 +02:00
{
2014-10-07 02:22:47 +02:00
DoPush ( ) ;
2015-05-10 14:48:35 +02:00
UniValue array ( UniValue : : VARR ) ;
2014-10-07 02:22:47 +02:00
array . push_back ( FormatScript ( spendTx . vin [ 0 ] . scriptSig ) ) ;
array . push_back ( FormatScript ( creditTx . vout [ 0 ] . scriptPubKey ) ) ;
array . push_back ( FormatScriptFlags ( flags ) ) ;
array . push_back ( comment ) ;
return array ;
2014-09-20 03:13:04 +02:00
}
std : : string GetComment ( )
{
return comment ;
}
const CScript & GetScriptPubKey ( )
{
return creditTx . vout [ 0 ] . scriptPubKey ;
}
} ;
}
BOOST_AUTO_TEST_CASE ( script_build )
{
2014-11-05 18:43:44 +01:00
const KeyData keys ;
2014-09-20 03:13:04 +02:00
std : : vector < TestBuilder > good ;
std : : vector < TestBuilder > bad ;
2014-09-25 04:54:08 +02:00
good . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey0 ) < < OP_CHECKSIG ,
2014-09-20 03:13:04 +02:00
" P2PK " , 0
) . PushSig ( keys . key0 ) ) ;
2014-09-25 04:54:08 +02:00
bad . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey0 ) < < OP_CHECKSIG ,
2014-09-20 03:13:04 +02:00
" P2PK, bad sig " , 0
) . PushSig ( keys . key0 ) . DamagePush ( 10 ) ) ;
2014-09-25 04:54:08 +02:00
good . push_back ( TestBuilder ( CScript ( ) < < OP_DUP < < OP_HASH160 < < ToByteVector ( keys . pubkey1C . GetID ( ) ) < < OP_EQUALVERIFY < < OP_CHECKSIG ,
2014-09-20 03:13:04 +02:00
" P2PKH " , 0
) . PushSig ( keys . key1 ) . Push ( keys . pubkey1C ) ) ;
2014-09-25 04:54:08 +02:00
bad . push_back ( TestBuilder ( CScript ( ) < < OP_DUP < < OP_HASH160 < < ToByteVector ( keys . pubkey2C . GetID ( ) ) < < OP_EQUALVERIFY < < OP_CHECKSIG ,
2014-09-20 03:13:04 +02:00
" P2PKH, bad pubkey " , 0
) . PushSig ( keys . key2 ) . Push ( keys . pubkey2C ) . DamagePush ( 5 ) ) ;
2014-09-25 04:54:08 +02:00
good . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey1 ) < < OP_CHECKSIG ,
2014-09-20 03:13:04 +02:00
" P2PK anyonecanpay " , 0
) . PushSig ( keys . key1 , SIGHASH_ALL | SIGHASH_ANYONECANPAY ) ) ;
2014-09-25 04:54:08 +02:00
bad . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey1 ) < < OP_CHECKSIG ,
2014-09-20 03:13:04 +02:00
" P2PK anyonecanpay marked with normal hashtype " , 0
) . PushSig ( keys . key1 , SIGHASH_ALL | SIGHASH_ANYONECANPAY ) . EditPush ( 70 , " 81 " , " 01 " ) ) ;
2014-09-25 04:54:08 +02:00
good . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey0C ) < < OP_CHECKSIG ,
2014-09-20 03:13:04 +02:00
" P2SH(P2PK) " , SCRIPT_VERIFY_P2SH , true
) . PushSig ( keys . key0 ) . PushRedeem ( ) ) ;
2014-09-25 04:54:08 +02:00
bad . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey0C ) < < OP_CHECKSIG ,
2014-09-20 03:13:04 +02:00
" P2SH(P2PK), bad redeemscript " , SCRIPT_VERIFY_P2SH , true
) . PushSig ( keys . key0 ) . PushRedeem ( ) . DamagePush ( 10 ) ) ;
2014-09-25 04:54:08 +02:00
good . push_back ( TestBuilder ( CScript ( ) < < OP_DUP < < OP_HASH160 < < ToByteVector ( keys . pubkey1 . GetID ( ) ) < < OP_EQUALVERIFY < < OP_CHECKSIG ,
2014-09-20 03:13:04 +02:00
" P2SH(P2PKH), bad sig but no VERIFY_P2SH " , 0 , true
) . PushSig ( keys . key0 ) . DamagePush ( 10 ) . PushRedeem ( ) ) ;
2014-09-25 04:54:08 +02:00
bad . push_back ( TestBuilder ( CScript ( ) < < OP_DUP < < OP_HASH160 < < ToByteVector ( keys . pubkey1 . GetID ( ) ) < < OP_EQUALVERIFY < < OP_CHECKSIG ,
2014-09-20 03:13:04 +02:00
" P2SH(P2PKH), bad sig " , SCRIPT_VERIFY_P2SH , true
) . PushSig ( keys . key0 ) . DamagePush ( 10 ) . PushRedeem ( ) ) ;
2014-09-25 04:54:08 +02:00
good . push_back ( TestBuilder ( CScript ( ) < < OP_3 < < ToByteVector ( keys . pubkey0C ) < < ToByteVector ( keys . pubkey1C ) < < ToByteVector ( keys . pubkey2C ) < < OP_3 < < OP_CHECKMULTISIG ,
2014-09-20 03:13:04 +02:00
" 3-of-3 " , 0
) . Num ( 0 ) . PushSig ( keys . key0 ) . PushSig ( keys . key1 ) . PushSig ( keys . key2 ) ) ;
2014-09-25 04:54:08 +02:00
bad . push_back ( TestBuilder ( CScript ( ) < < OP_3 < < ToByteVector ( keys . pubkey0C ) < < ToByteVector ( keys . pubkey1C ) < < ToByteVector ( keys . pubkey2C ) < < OP_3 < < OP_CHECKMULTISIG ,
2014-09-20 03:13:04 +02:00
" 3-of-3, 2 sigs " , 0
) . Num ( 0 ) . PushSig ( keys . key0 ) . PushSig ( keys . key1 ) . Num ( 0 ) ) ;
2014-09-25 04:54:08 +02:00
good . push_back ( TestBuilder ( CScript ( ) < < OP_2 < < ToByteVector ( keys . pubkey0C ) < < ToByteVector ( keys . pubkey1C ) < < ToByteVector ( keys . pubkey2C ) < < OP_3 < < OP_CHECKMULTISIG ,
2014-09-20 03:13:04 +02:00
" P2SH(2-of-3) " , SCRIPT_VERIFY_P2SH , true
) . Num ( 0 ) . PushSig ( keys . key1 ) . PushSig ( keys . key2 ) . PushRedeem ( ) ) ;
2014-09-25 04:54:08 +02:00
bad . push_back ( TestBuilder ( CScript ( ) < < OP_2 < < ToByteVector ( keys . pubkey0C ) < < ToByteVector ( keys . pubkey1C ) < < ToByteVector ( keys . pubkey2C ) < < OP_3 < < OP_CHECKMULTISIG ,
2014-09-20 03:13:04 +02:00
" P2SH(2-of-3), 1 sig " , SCRIPT_VERIFY_P2SH , true
) . Num ( 0 ) . PushSig ( keys . key1 ) . Num ( 0 ) . PushRedeem ( ) ) ;
2014-09-25 04:54:08 +02:00
good . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey1C ) < < OP_CHECKSIG ,
2014-10-07 02:22:47 +02:00
" P2PK with too much R padding but no DERSIG " , 0
2014-09-20 03:13:04 +02:00
) . PushSig ( keys . key1 , SIGHASH_ALL , 31 , 32 ) . EditPush ( 1 , " 43021F " , " 44022000 " ) ) ;
2014-09-25 04:54:08 +02:00
bad . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey1C ) < < OP_CHECKSIG ,
2014-10-07 02:22:47 +02:00
" P2PK with too much R padding " , SCRIPT_VERIFY_DERSIG
2014-09-20 03:13:04 +02:00
) . PushSig ( keys . key1 , SIGHASH_ALL , 31 , 32 ) . EditPush ( 1 , " 43021F " , " 44022000 " ) ) ;
2014-09-25 04:54:08 +02:00
good . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey1C ) < < OP_CHECKSIG ,
2014-10-07 02:22:47 +02:00
" P2PK with too much S padding but no DERSIG " , 0
2014-09-20 03:13:04 +02:00
) . PushSig ( keys . key1 , SIGHASH_ALL ) . EditPush ( 1 , " 44 " , " 45 " ) . EditPush ( 37 , " 20 " , " 2100 " ) ) ;
2014-09-25 04:54:08 +02:00
bad . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey1C ) < < OP_CHECKSIG ,
2014-10-07 02:22:47 +02:00
" P2PK with too much S padding " , SCRIPT_VERIFY_DERSIG
2014-09-20 03:13:04 +02:00
) . PushSig ( keys . key1 , SIGHASH_ALL ) . EditPush ( 1 , " 44 " , " 45 " ) . EditPush ( 37 , " 20 " , " 2100 " ) ) ;
2014-09-25 04:54:08 +02:00
good . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey1C ) < < OP_CHECKSIG ,
2014-10-07 02:22:47 +02:00
" P2PK with too little R padding but no DERSIG " , 0
2014-09-20 03:13:04 +02:00
) . PushSig ( keys . key1 , SIGHASH_ALL , 33 , 32 ) . EditPush ( 1 , " 45022100 " , " 440220 " ) ) ;
2014-09-25 04:54:08 +02:00
bad . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey1C ) < < OP_CHECKSIG ,
2014-10-07 02:22:47 +02:00
" P2PK with too little R padding " , SCRIPT_VERIFY_DERSIG
2014-09-20 03:13:04 +02:00
) . PushSig ( keys . key1 , SIGHASH_ALL , 33 , 32 ) . EditPush ( 1 , " 45022100 " , " 440220 " ) ) ;
2014-09-25 04:54:08 +02:00
good . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey2C ) < < OP_CHECKSIG < < OP_NOT ,
2014-10-07 02:22:47 +02:00
" P2PK NOT with bad sig with too much R padding but no DERSIG " , 0
) . PushSig ( keys . key2 , SIGHASH_ALL , 31 , 32 ) . EditPush ( 1 , " 43021F " , " 44022000 " ) . DamagePush ( 10 ) ) ;
2014-09-25 04:54:08 +02:00
bad . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey2C ) < < OP_CHECKSIG < < OP_NOT ,
2014-10-07 02:22:47 +02:00
" P2PK NOT with bad sig with too much R padding " , SCRIPT_VERIFY_DERSIG
) . PushSig ( keys . key2 , SIGHASH_ALL , 31 , 32 ) . EditPush ( 1 , " 43021F " , " 44022000 " ) . DamagePush ( 10 ) ) ;
2014-09-25 04:54:08 +02:00
bad . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey2C ) < < OP_CHECKSIG < < OP_NOT ,
2014-10-07 02:22:47 +02:00
" P2PK NOT with too much R padding but no DERSIG " , 0
) . PushSig ( keys . key2 , SIGHASH_ALL , 31 , 32 ) . EditPush ( 1 , " 43021F " , " 44022000 " ) ) ;
2014-09-25 04:54:08 +02:00
bad . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey2C ) < < OP_CHECKSIG < < OP_NOT ,
2014-10-07 02:22:47 +02:00
" P2PK NOT with too much R padding " , SCRIPT_VERIFY_DERSIG
) . PushSig ( keys . key2 , SIGHASH_ALL , 31 , 32 ) . EditPush ( 1 , " 43021F " , " 44022000 " ) ) ;
2014-09-20 03:13:04 +02:00
2015-01-20 00:19:37 +01:00
good . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey1C ) < < OP_CHECKSIG ,
" BIP66 example 1, without DERSIG " , 0
) . PushSig ( keys . key1 , SIGHASH_ALL , 33 , 32 ) . EditPush ( 1 , " 45022100 " , " 440220 " ) ) ;
bad . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey1C ) < < OP_CHECKSIG ,
" BIP66 example 1, with DERSIG " , SCRIPT_VERIFY_DERSIG
) . PushSig ( keys . key1 , SIGHASH_ALL , 33 , 32 ) . EditPush ( 1 , " 45022100 " , " 440220 " ) ) ;
bad . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey1C ) < < OP_CHECKSIG < < OP_NOT ,
" BIP66 example 2, without DERSIG " , 0
) . PushSig ( keys . key1 , SIGHASH_ALL , 33 , 32 ) . EditPush ( 1 , " 45022100 " , " 440220 " ) ) ;
bad . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey1C ) < < OP_CHECKSIG < < OP_NOT ,
" BIP66 example 2, with DERSIG " , SCRIPT_VERIFY_DERSIG
) . PushSig ( keys . key1 , SIGHASH_ALL , 33 , 32 ) . EditPush ( 1 , " 45022100 " , " 440220 " ) ) ;
bad . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey1C ) < < OP_CHECKSIG ,
" BIP66 example 3, without DERSIG " , 0
) . Num ( 0 ) ) ;
bad . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey1C ) < < OP_CHECKSIG ,
" BIP66 example 3, with DERSIG " , SCRIPT_VERIFY_DERSIG
) . Num ( 0 ) ) ;
good . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey1C ) < < OP_CHECKSIG < < OP_NOT ,
" BIP66 example 4, without DERSIG " , 0
) . Num ( 0 ) ) ;
good . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey1C ) < < OP_CHECKSIG < < OP_NOT ,
" BIP66 example 4, with DERSIG " , SCRIPT_VERIFY_DERSIG
) . Num ( 0 ) ) ;
bad . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey1C ) < < OP_CHECKSIG ,
" BIP66 example 5, without DERSIG " , 0
) . Num ( 1 ) ) ;
bad . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey1C ) < < OP_CHECKSIG ,
" BIP66 example 5, with DERSIG " , SCRIPT_VERIFY_DERSIG
) . Num ( 1 ) ) ;
good . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey1C ) < < OP_CHECKSIG < < OP_NOT ,
" BIP66 example 6, without DERSIG " , 0
) . Num ( 1 ) ) ;
bad . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey1C ) < < OP_CHECKSIG < < OP_NOT ,
" BIP66 example 6, with DERSIG " , SCRIPT_VERIFY_DERSIG
) . Num ( 1 ) ) ;
good . push_back ( TestBuilder ( CScript ( ) < < OP_2 < < ToByteVector ( keys . pubkey1C ) < < ToByteVector ( keys . pubkey2C ) < < OP_2 < < OP_CHECKMULTISIG ,
" BIP66 example 7, without DERSIG " , 0
) . Num ( 0 ) . PushSig ( keys . key1 , SIGHASH_ALL , 33 , 32 ) . EditPush ( 1 , " 45022100 " , " 440220 " ) . PushSig ( keys . key2 ) ) ;
bad . push_back ( TestBuilder ( CScript ( ) < < OP_2 < < ToByteVector ( keys . pubkey1C ) < < ToByteVector ( keys . pubkey2C ) < < OP_2 < < OP_CHECKMULTISIG ,
" BIP66 example 7, with DERSIG " , SCRIPT_VERIFY_DERSIG
) . Num ( 0 ) . PushSig ( keys . key1 , SIGHASH_ALL , 33 , 32 ) . EditPush ( 1 , " 45022100 " , " 440220 " ) . PushSig ( keys . key2 ) ) ;
bad . push_back ( TestBuilder ( CScript ( ) < < OP_2 < < ToByteVector ( keys . pubkey1C ) < < ToByteVector ( keys . pubkey2C ) < < OP_2 < < OP_CHECKMULTISIG < < OP_NOT ,
" BIP66 example 8, without DERSIG " , 0
) . Num ( 0 ) . PushSig ( keys . key1 , SIGHASH_ALL , 33 , 32 ) . EditPush ( 1 , " 45022100 " , " 440220 " ) . PushSig ( keys . key2 ) ) ;
bad . push_back ( TestBuilder ( CScript ( ) < < OP_2 < < ToByteVector ( keys . pubkey1C ) < < ToByteVector ( keys . pubkey2C ) < < OP_2 < < OP_CHECKMULTISIG < < OP_NOT ,
" BIP66 example 8, with DERSIG " , SCRIPT_VERIFY_DERSIG
) . Num ( 0 ) . PushSig ( keys . key1 , SIGHASH_ALL , 33 , 32 ) . EditPush ( 1 , " 45022100 " , " 440220 " ) . PushSig ( keys . key2 ) ) ;
bad . push_back ( TestBuilder ( CScript ( ) < < OP_2 < < ToByteVector ( keys . pubkey1C ) < < ToByteVector ( keys . pubkey2C ) < < OP_2 < < OP_CHECKMULTISIG ,
" BIP66 example 9, without DERSIG " , 0
) . Num ( 0 ) . Num ( 0 ) . PushSig ( keys . key2 , SIGHASH_ALL , 33 , 32 ) . EditPush ( 1 , " 45022100 " , " 440220 " ) ) ;
bad . push_back ( TestBuilder ( CScript ( ) < < OP_2 < < ToByteVector ( keys . pubkey1C ) < < ToByteVector ( keys . pubkey2C ) < < OP_2 < < OP_CHECKMULTISIG ,
" BIP66 example 9, with DERSIG " , SCRIPT_VERIFY_DERSIG
) . Num ( 0 ) . Num ( 0 ) . PushSig ( keys . key2 , SIGHASH_ALL , 33 , 32 ) . EditPush ( 1 , " 45022100 " , " 440220 " ) ) ;
good . push_back ( TestBuilder ( CScript ( ) < < OP_2 < < ToByteVector ( keys . pubkey1C ) < < ToByteVector ( keys . pubkey2C ) < < OP_2 < < OP_CHECKMULTISIG < < OP_NOT ,
" BIP66 example 10, without DERSIG " , 0
) . Num ( 0 ) . Num ( 0 ) . PushSig ( keys . key2 , SIGHASH_ALL , 33 , 32 ) . EditPush ( 1 , " 45022100 " , " 440220 " ) ) ;
bad . push_back ( TestBuilder ( CScript ( ) < < OP_2 < < ToByteVector ( keys . pubkey1C ) < < ToByteVector ( keys . pubkey2C ) < < OP_2 < < OP_CHECKMULTISIG < < OP_NOT ,
" BIP66 example 10, with DERSIG " , SCRIPT_VERIFY_DERSIG
) . Num ( 0 ) . Num ( 0 ) . PushSig ( keys . key2 , SIGHASH_ALL , 33 , 32 ) . EditPush ( 1 , " 45022100 " , " 440220 " ) ) ;
bad . push_back ( TestBuilder ( CScript ( ) < < OP_2 < < ToByteVector ( keys . pubkey1C ) < < ToByteVector ( keys . pubkey2C ) < < OP_2 < < OP_CHECKMULTISIG ,
" BIP66 example 11, without DERSIG " , 0
) . Num ( 0 ) . PushSig ( keys . key1 , SIGHASH_ALL , 33 , 32 ) . EditPush ( 1 , " 45022100 " , " 440220 " ) . Num ( 0 ) ) ;
bad . push_back ( TestBuilder ( CScript ( ) < < OP_2 < < ToByteVector ( keys . pubkey1C ) < < ToByteVector ( keys . pubkey2C ) < < OP_2 < < OP_CHECKMULTISIG ,
" BIP66 example 11, with DERSIG " , SCRIPT_VERIFY_DERSIG
) . Num ( 0 ) . PushSig ( keys . key1 , SIGHASH_ALL , 33 , 32 ) . EditPush ( 1 , " 45022100 " , " 440220 " ) . Num ( 0 ) ) ;
good . push_back ( TestBuilder ( CScript ( ) < < OP_2 < < ToByteVector ( keys . pubkey1C ) < < ToByteVector ( keys . pubkey2C ) < < OP_2 < < OP_CHECKMULTISIG < < OP_NOT ,
" BIP66 example 12, without DERSIG " , 0
) . Num ( 0 ) . PushSig ( keys . key1 , SIGHASH_ALL , 33 , 32 ) . EditPush ( 1 , " 45022100 " , " 440220 " ) . Num ( 0 ) ) ;
good . push_back ( TestBuilder ( CScript ( ) < < OP_2 < < ToByteVector ( keys . pubkey1C ) < < ToByteVector ( keys . pubkey2C ) < < OP_2 < < OP_CHECKMULTISIG < < OP_NOT ,
" BIP66 example 12, with DERSIG " , SCRIPT_VERIFY_DERSIG
) . Num ( 0 ) . PushSig ( keys . key1 , SIGHASH_ALL , 33 , 32 ) . EditPush ( 1 , " 45022100 " , " 440220 " ) . Num ( 0 ) ) ;
2015-02-10 18:11:59 +01:00
good . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey2C ) < < OP_CHECKSIG ,
" P2PK with multi-byte hashtype, without DERSIG " , 0
) . PushSig ( keys . key2 , SIGHASH_ALL ) . EditPush ( 70 , " 01 " , " 0101 " ) ) ;
bad . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey2C ) < < OP_CHECKSIG ,
" P2PK with multi-byte hashtype, with DERSIG " , SCRIPT_VERIFY_DERSIG
) . PushSig ( keys . key2 , SIGHASH_ALL ) . EditPush ( 70 , " 01 " , " 0101 " ) ) ;
2015-01-20 00:19:37 +01:00
2014-09-25 04:54:08 +02:00
good . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey2C ) < < OP_CHECKSIG ,
2014-10-07 02:22:47 +02:00
" P2PK with high S but no LOW_S " , 0
2014-09-20 03:13:04 +02:00
) . PushSig ( keys . key2 , SIGHASH_ALL , 32 , 33 ) ) ;
2014-09-25 04:54:08 +02:00
bad . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey2C ) < < OP_CHECKSIG ,
2014-10-07 02:22:47 +02:00
" P2PK with high S " , SCRIPT_VERIFY_LOW_S
2014-09-20 03:13:04 +02:00
) . PushSig ( keys . key2 , SIGHASH_ALL , 32 , 33 ) ) ;
2014-09-25 04:54:08 +02:00
good . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey0H ) < < OP_CHECKSIG ,
2014-10-07 02:22:47 +02:00
" P2PK with hybrid pubkey but no STRICTENC " , 0
) . PushSig ( keys . key0 , SIGHASH_ALL ) ) ;
2014-09-25 04:54:08 +02:00
bad . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey0H ) < < OP_CHECKSIG ,
2014-10-07 02:22:47 +02:00
" P2PK with hybrid pubkey " , SCRIPT_VERIFY_STRICTENC
) . PushSig ( keys . key0 , SIGHASH_ALL ) ) ;
2014-09-25 04:54:08 +02:00
bad . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey0H ) < < OP_CHECKSIG < < OP_NOT ,
2014-10-07 02:22:47 +02:00
" P2PK NOT with hybrid pubkey but no STRICTENC " , 0
) . PushSig ( keys . key0 , SIGHASH_ALL ) ) ;
2014-11-08 18:32:29 +01:00
bad . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey0H ) < < OP_CHECKSIG < < OP_NOT ,
" P2PK NOT with hybrid pubkey " , SCRIPT_VERIFY_STRICTENC
) . PushSig ( keys . key0 , SIGHASH_ALL ) ) ;
2014-09-25 04:54:08 +02:00
good . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey0H ) < < OP_CHECKSIG < < OP_NOT ,
2014-10-07 02:22:47 +02:00
" P2PK NOT with invalid hybrid pubkey but no STRICTENC " , 0
) . PushSig ( keys . key0 , SIGHASH_ALL ) . DamagePush ( 10 ) ) ;
2014-11-08 18:32:29 +01:00
bad . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey0H ) < < OP_CHECKSIG < < OP_NOT ,
" P2PK NOT with invalid hybrid pubkey " , SCRIPT_VERIFY_STRICTENC
) . PushSig ( keys . key0 , SIGHASH_ALL ) . DamagePush ( 10 ) ) ;
good . push_back ( TestBuilder ( CScript ( ) < < OP_1 < < ToByteVector ( keys . pubkey0H ) < < ToByteVector ( keys . pubkey1C ) < < OP_2 < < OP_CHECKMULTISIG ,
" 1-of-2 with the second 1 hybrid pubkey and no STRICTENC " , 0
) . Num ( 0 ) . PushSig ( keys . key1 , SIGHASH_ALL ) ) ;
good . push_back ( TestBuilder ( CScript ( ) < < OP_1 < < ToByteVector ( keys . pubkey0H ) < < ToByteVector ( keys . pubkey1C ) < < OP_2 < < OP_CHECKMULTISIG ,
" 1-of-2 with the second 1 hybrid pubkey " , SCRIPT_VERIFY_STRICTENC
) . Num ( 0 ) . PushSig ( keys . key1 , SIGHASH_ALL ) ) ;
bad . push_back ( TestBuilder ( CScript ( ) < < OP_1 < < ToByteVector ( keys . pubkey1C ) < < ToByteVector ( keys . pubkey0H ) < < OP_2 < < OP_CHECKMULTISIG ,
" 1-of-2 with the first 1 hybrid pubkey " , SCRIPT_VERIFY_STRICTENC
) . Num ( 0 ) . PushSig ( keys . key1 , SIGHASH_ALL ) ) ;
2014-10-07 02:22:47 +02:00
2014-09-25 04:54:08 +02:00
good . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey1 ) < < OP_CHECKSIG ,
2014-10-07 02:22:47 +02:00
" P2PK with undefined hashtype but no STRICTENC " , 0
) . PushSig ( keys . key1 , 5 ) ) ;
2014-09-25 04:54:08 +02:00
bad . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey1 ) < < OP_CHECKSIG ,
2014-10-07 02:22:47 +02:00
" P2PK with undefined hashtype " , SCRIPT_VERIFY_STRICTENC
) . PushSig ( keys . key1 , 5 ) ) ;
2014-09-25 04:54:08 +02:00
good . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey1 ) < < OP_CHECKSIG < < OP_NOT ,
2014-10-07 02:22:47 +02:00
" P2PK NOT with invalid sig and undefined hashtype but no STRICTENC " , 0
) . PushSig ( keys . key1 , 5 ) . DamagePush ( 10 ) ) ;
2014-09-25 04:54:08 +02:00
bad . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey1 ) < < OP_CHECKSIG < < OP_NOT ,
2014-10-07 02:22:47 +02:00
" P2PK NOT with invalid sig and undefined hashtype " , SCRIPT_VERIFY_STRICTENC
) . PushSig ( keys . key1 , 5 ) . DamagePush ( 10 ) ) ;
2014-09-25 04:54:08 +02:00
good . push_back ( TestBuilder ( CScript ( ) < < OP_3 < < ToByteVector ( keys . pubkey0C ) < < ToByteVector ( keys . pubkey1C ) < < ToByteVector ( keys . pubkey2C ) < < OP_3 < < OP_CHECKMULTISIG ,
2014-09-20 03:13:04 +02:00
" 3-of-3 with nonzero dummy but no NULLDUMMY " , 0
) . Num ( 1 ) . PushSig ( keys . key0 ) . PushSig ( keys . key1 ) . PushSig ( keys . key2 ) ) ;
2014-09-25 04:54:08 +02:00
bad . push_back ( TestBuilder ( CScript ( ) < < OP_3 < < ToByteVector ( keys . pubkey0C ) < < ToByteVector ( keys . pubkey1C ) < < ToByteVector ( keys . pubkey2C ) < < OP_3 < < OP_CHECKMULTISIG ,
2014-09-20 03:13:04 +02:00
" 3-of-3 with nonzero dummy " , SCRIPT_VERIFY_NULLDUMMY
) . Num ( 1 ) . PushSig ( keys . key0 ) . PushSig ( keys . key1 ) . PushSig ( keys . key2 ) ) ;
2014-09-25 04:54:08 +02:00
good . push_back ( TestBuilder ( CScript ( ) < < OP_3 < < ToByteVector ( keys . pubkey0C ) < < ToByteVector ( keys . pubkey1C ) < < ToByteVector ( keys . pubkey2C ) < < OP_3 < < OP_CHECKMULTISIG < < OP_NOT ,
2014-10-07 02:22:47 +02:00
" 3-of-3 NOT with invalid sig and nonzero dummy but no NULLDUMMY " , 0
) . Num ( 1 ) . PushSig ( keys . key0 ) . PushSig ( keys . key1 ) . PushSig ( keys . key2 ) . DamagePush ( 10 ) ) ;
2014-09-25 04:54:08 +02:00
bad . push_back ( TestBuilder ( CScript ( ) < < OP_3 < < ToByteVector ( keys . pubkey0C ) < < ToByteVector ( keys . pubkey1C ) < < ToByteVector ( keys . pubkey2C ) < < OP_3 < < OP_CHECKMULTISIG < < OP_NOT ,
2014-10-07 02:22:47 +02:00
" 3-of-3 NOT with invalid sig with nonzero dummy " , SCRIPT_VERIFY_NULLDUMMY
) . Num ( 1 ) . PushSig ( keys . key0 ) . PushSig ( keys . key1 ) . PushSig ( keys . key2 ) . DamagePush ( 10 ) ) ;
2014-09-20 03:13:04 +02:00
2014-10-09 01:29:45 +02:00
good . push_back ( TestBuilder ( CScript ( ) < < OP_2 < < ToByteVector ( keys . pubkey1C ) < < ToByteVector ( keys . pubkey1C ) < < OP_2 < < OP_CHECKMULTISIG ,
" 2-of-2 with two identical keys and sigs pushed using OP_DUP but no SIGPUSHONLY " , 0
) . Num ( 0 ) . PushSig ( keys . key1 ) . Add ( CScript ( ) < < OP_DUP ) ) ;
bad . push_back ( TestBuilder ( CScript ( ) < < OP_2 < < ToByteVector ( keys . pubkey1C ) < < ToByteVector ( keys . pubkey1C ) < < OP_2 < < OP_CHECKMULTISIG ,
" 2-of-2 with two identical keys and sigs pushed using OP_DUP " , SCRIPT_VERIFY_SIGPUSHONLY
) . Num ( 0 ) . PushSig ( keys . key1 ) . Add ( CScript ( ) < < OP_DUP ) ) ;
bad . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey2C ) < < OP_CHECKSIG ,
" P2SH(P2PK) with non-push scriptSig but no SIGPUSHONLY " , 0
) . PushSig ( keys . key2 ) . PushRedeem ( ) ) ;
bad . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey2C ) < < OP_CHECKSIG ,
" P2SH(P2PK) with non-push scriptSig " , SCRIPT_VERIFY_SIGPUSHONLY
) . PushSig ( keys . key2 ) . PushRedeem ( ) ) ;
good . push_back ( TestBuilder ( CScript ( ) < < OP_2 < < ToByteVector ( keys . pubkey1C ) < < ToByteVector ( keys . pubkey1C ) < < OP_2 < < OP_CHECKMULTISIG ,
" 2-of-2 with two identical keys and sigs pushed " , SCRIPT_VERIFY_SIGPUSHONLY
) . Num ( 0 ) . PushSig ( keys . key1 ) . PushSig ( keys . key1 ) ) ;
2014-10-13 03:39:47 +02:00
good . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey0 ) < < OP_CHECKSIG ,
" P2PK with unnecessary input but no CLEANSTACK " , SCRIPT_VERIFY_P2SH
) . Num ( 11 ) . PushSig ( keys . key0 ) ) ;
bad . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey0 ) < < OP_CHECKSIG ,
" P2PK with unnecessary input " , SCRIPT_VERIFY_CLEANSTACK | SCRIPT_VERIFY_P2SH
) . Num ( 11 ) . PushSig ( keys . key0 ) ) ;
good . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey0 ) < < OP_CHECKSIG ,
" P2SH with unnecessary input but no CLEANSTACK " , SCRIPT_VERIFY_P2SH , true
) . Num ( 11 ) . PushSig ( keys . key0 ) . PushRedeem ( ) ) ;
bad . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey0 ) < < OP_CHECKSIG ,
" P2SH with unnecessary input " , SCRIPT_VERIFY_CLEANSTACK | SCRIPT_VERIFY_P2SH , true
) . Num ( 11 ) . PushSig ( keys . key0 ) . PushRedeem ( ) ) ;
good . push_back ( TestBuilder ( CScript ( ) < < ToByteVector ( keys . pubkey0 ) < < OP_CHECKSIG ,
" P2SH with CLEANSTACK " , SCRIPT_VERIFY_CLEANSTACK | SCRIPT_VERIFY_P2SH , true
) . PushSig ( keys . key0 ) . PushRedeem ( ) ) ;
2014-10-09 01:29:45 +02:00
2014-12-01 21:26:51 +01:00
std : : set < std : : string > tests_good ;
std : : set < std : : string > tests_bad ;
2014-09-20 03:13:04 +02:00
{
Array json_good = read_json ( std : : string ( json_tests : : script_valid , json_tests : : script_valid + sizeof ( json_tests : : script_valid ) ) ) ;
Array json_bad = read_json ( std : : string ( json_tests : : script_invalid , json_tests : : script_invalid + sizeof ( json_tests : : script_invalid ) ) ) ;
2015-05-10 13:35:44 +02:00
for ( unsigned int idx = 0 ; idx < json_good . size ( ) ; idx + + ) {
const Value & tv = json_good [ idx ] ;
tests_good . insert ( tv . get_array ( ) . write ( ) ) ;
2014-09-20 03:13:04 +02:00
}
2015-05-10 13:35:44 +02:00
for ( unsigned int idx = 0 ; idx < json_bad . size ( ) ; idx + + ) {
const Value & tv = json_bad [ idx ] ;
tests_bad . insert ( tv . get_array ( ) . write ( ) ) ;
2014-09-20 03:13:04 +02:00
}
}
std : : string strGood ;
std : : string strBad ;
BOOST_FOREACH ( TestBuilder & test , good ) {
test . Test ( true ) ;
2015-05-10 13:35:44 +02:00
std : : string str = test . GetJSON ( ) . write ( ) ;
2014-10-07 02:22:47 +02:00
# ifndef UPDATE_JSON_TESTS
2014-12-01 21:26:51 +01:00
if ( tests_good . count ( str ) = = 0 ) {
2014-10-07 02:22:47 +02:00
BOOST_CHECK_MESSAGE ( false , " Missing auto script_valid test: " + test . GetComment ( ) ) ;
}
2014-12-01 21:26:51 +01:00
# endif
strGood + = str + " , \n " ;
2014-09-20 03:13:04 +02:00
}
BOOST_FOREACH ( TestBuilder & test , bad ) {
test . Test ( false ) ;
2015-05-10 13:35:44 +02:00
std : : string str = test . GetJSON ( ) . write ( ) ;
2014-10-07 02:22:47 +02:00
# ifndef UPDATE_JSON_TESTS
2014-12-01 21:26:51 +01:00
if ( tests_bad . count ( str ) = = 0 ) {
2014-10-07 02:22:47 +02:00
BOOST_CHECK_MESSAGE ( false , " Missing auto script_invalid test: " + test . GetComment ( ) ) ;
}
2014-12-01 21:26:51 +01:00
# endif
strBad + = str + " , \n " ;
2014-09-20 03:13:04 +02:00
}
2014-10-07 02:22:47 +02:00
# ifdef UPDATE_JSON_TESTS
2014-09-20 03:13:04 +02:00
FILE * valid = fopen ( " script_valid.json.gen " , " w " ) ;
fputs ( strGood . c_str ( ) , valid ) ;
fclose ( valid ) ;
FILE * invalid = fopen ( " script_invalid.json.gen " , " w " ) ;
fputs ( strBad . c_str ( ) , invalid ) ;
fclose ( invalid ) ;
# endif
}
2012-04-17 23:57:06 +02:00
BOOST_AUTO_TEST_CASE ( script_valid )
{
// Read tests from test/data/script_valid.json
// Format is an array of arrays
2014-09-08 21:15:31 +02:00
// Inner arrays are [ "scriptSig", "scriptPubKey", "flags" ]
2012-04-17 23:57:06 +02:00
// ... where scriptSig and scriptPubKey are stringified
// scripts.
2013-09-10 21:18:09 +02:00
Array tests = read_json ( std : : string ( json_tests : : script_valid , json_tests : : script_valid + sizeof ( json_tests : : script_valid ) ) ) ;
2012-04-17 23:57:06 +02:00
2014-08-20 21:15:16 +02:00
for ( unsigned int idx = 0 ; idx < tests . size ( ) ; idx + + ) {
Array test = tests [ idx ] ;
string strTest = test . write ( ) ;
2014-09-08 21:15:31 +02:00
if ( test . size ( ) < 3 ) // Allow size > 3; extra stuff ignored (useful for comments)
2012-04-17 23:57:06 +02:00
{
2014-09-20 00:20:53 +02:00
if ( test . size ( ) ! = 1 ) {
BOOST_ERROR ( " Bad test: " < < strTest ) ;
}
2012-04-17 23:57:06 +02:00
continue ;
}
string scriptSigString = test [ 0 ] . get_str ( ) ;
CScript scriptSig = ParseScript ( scriptSigString ) ;
string scriptPubKeyString = test [ 1 ] . get_str ( ) ;
CScript scriptPubKey = ParseScript ( scriptPubKeyString ) ;
2014-09-08 21:15:31 +02:00
unsigned int scriptflags = ParseScriptFlags ( test [ 2 ] . get_str ( ) ) ;
2012-04-17 23:57:06 +02:00
2014-09-20 03:13:04 +02:00
DoTest ( scriptPubKey , scriptSig , scriptflags , true , strTest ) ;
2012-04-17 23:57:06 +02:00
}
}
BOOST_AUTO_TEST_CASE ( script_invalid )
{
// Scripts that should evaluate as invalid
2013-09-10 21:18:09 +02:00
Array tests = read_json ( std : : string ( json_tests : : script_invalid , json_tests : : script_invalid + sizeof ( json_tests : : script_invalid ) ) ) ;
2012-04-17 23:57:06 +02:00
2014-08-20 21:15:16 +02:00
for ( unsigned int idx = 0 ; idx < tests . size ( ) ; idx + + ) {
Array test = tests [ idx ] ;
string strTest = test . write ( ) ;
if ( test . size ( ) < 3 ) // Allow size > 2; extra stuff ignored (useful for comments)
2012-04-17 23:57:06 +02:00
{
2014-09-20 00:20:53 +02:00
if ( test . size ( ) ! = 1 ) {
BOOST_ERROR ( " Bad test: " < < strTest ) ;
}
2012-04-17 23:57:06 +02:00
continue ;
}
string scriptSigString = test [ 0 ] . get_str ( ) ;
CScript scriptSig = ParseScript ( scriptSigString ) ;
string scriptPubKeyString = test [ 1 ] . get_str ( ) ;
CScript scriptPubKey = ParseScript ( scriptPubKeyString ) ;
2014-09-08 21:15:31 +02:00
unsigned int scriptflags = ParseScriptFlags ( test [ 2 ] . get_str ( ) ) ;
2012-04-17 23:57:06 +02:00
2014-09-20 03:13:04 +02:00
DoTest ( scriptPubKey , scriptSig , scriptflags , false , strTest ) ;
2012-04-17 23:57:06 +02:00
}
}
2011-07-31 20:01:31 +02:00
BOOST_AUTO_TEST_CASE ( script_PushData )
{
// Check that PUSHDATA1, PUSHDATA2, and PUSHDATA4 create the same value on
// the stack as the 1-75 opcodes do.
static const unsigned char direct [ ] = { 1 , 0x5a } ;
static const unsigned char pushdata1 [ ] = { OP_PUSHDATA1 , 1 , 0x5a } ;
static const unsigned char pushdata2 [ ] = { OP_PUSHDATA2 , 1 , 0 , 0x5a } ;
static const unsigned char pushdata4 [ ] = { OP_PUSHDATA4 , 1 , 0 , 0 , 0 , 0x5a } ;
2014-11-13 20:27:38 +01:00
ScriptError err ;
2011-07-31 20:01:31 +02:00
vector < vector < unsigned char > > directStack ;
2014-12-29 16:48:19 +01:00
BOOST_CHECK ( EvalScript ( directStack , CScript ( & direct [ 0 ] , & direct [ sizeof ( direct ) ] ) , SCRIPT_VERIFY_P2SH , BaseSignatureChecker ( ) , & err ) ) ;
2014-11-13 20:27:38 +01:00
BOOST_CHECK_MESSAGE ( err = = SCRIPT_ERR_OK , ScriptErrorString ( err ) ) ;
2011-07-31 20:01:31 +02:00
vector < vector < unsigned char > > pushdata1Stack ;
2014-12-29 16:48:19 +01:00
BOOST_CHECK ( EvalScript ( pushdata1Stack , CScript ( & pushdata1 [ 0 ] , & pushdata1 [ sizeof ( pushdata1 ) ] ) , SCRIPT_VERIFY_P2SH , BaseSignatureChecker ( ) , & err ) ) ;
2011-07-31 20:01:31 +02:00
BOOST_CHECK ( pushdata1Stack = = directStack ) ;
2014-11-13 20:27:38 +01:00
BOOST_CHECK_MESSAGE ( err = = SCRIPT_ERR_OK , ScriptErrorString ( err ) ) ;
2011-07-31 20:01:31 +02:00
vector < vector < unsigned char > > pushdata2Stack ;
2014-12-29 16:48:19 +01:00
BOOST_CHECK ( EvalScript ( pushdata2Stack , CScript ( & pushdata2 [ 0 ] , & pushdata2 [ sizeof ( pushdata2 ) ] ) , SCRIPT_VERIFY_P2SH , BaseSignatureChecker ( ) , & err ) ) ;
2011-07-31 20:01:31 +02:00
BOOST_CHECK ( pushdata2Stack = = directStack ) ;
2014-11-13 20:27:38 +01:00
BOOST_CHECK_MESSAGE ( err = = SCRIPT_ERR_OK , ScriptErrorString ( err ) ) ;
2011-07-31 20:01:31 +02:00
vector < vector < unsigned char > > pushdata4Stack ;
2014-12-29 16:48:19 +01:00
BOOST_CHECK ( EvalScript ( pushdata4Stack , CScript ( & pushdata4 [ 0 ] , & pushdata4 [ sizeof ( pushdata4 ) ] ) , SCRIPT_VERIFY_P2SH , BaseSignatureChecker ( ) , & err ) ) ;
2011-07-31 20:01:31 +02:00
BOOST_CHECK ( pushdata4Stack = = directStack ) ;
2014-11-13 20:27:38 +01:00
BOOST_CHECK_MESSAGE ( err = = SCRIPT_ERR_OK , ScriptErrorString ( err ) ) ;
2011-07-31 20:01:31 +02:00
}
2011-08-27 00:41:22 +02:00
CScript
sign_multisig ( CScript scriptPubKey , std : : vector < CKey > keys , CTransaction transaction )
{
uint256 hash = SignatureHash ( scriptPubKey , transaction , 0 , SIGHASH_ALL ) ;
CScript result ;
//
// NOTE: CHECKMULTISIG has an unfortunate bug; it requires
// one extra item on the stack, before the signatures.
// Putting OP_0 on the stack is the workaround;
2012-10-05 19:22:21 +02:00
// fixing the bug would mean splitting the block chain (old
2011-08-27 00:41:22 +02:00
// clients would not accept new CHECKMULTISIG transactions,
// and vice-versa)
//
result < < OP_0 ;
2013-05-01 06:52:05 +02:00
BOOST_FOREACH ( const CKey & key , keys )
2011-08-27 00:41:22 +02:00
{
vector < unsigned char > vchSig ;
BOOST_CHECK ( key . Sign ( hash , vchSig ) ) ;
vchSig . push_back ( ( unsigned char ) SIGHASH_ALL ) ;
result < < vchSig ;
}
return result ;
}
CScript
2013-05-01 06:52:05 +02:00
sign_multisig ( CScript scriptPubKey , const CKey & key , CTransaction transaction )
2011-08-27 00:41:22 +02:00
{
std : : vector < CKey > keys ;
keys . push_back ( key ) ;
return sign_multisig ( scriptPubKey , keys , transaction ) ;
}
BOOST_AUTO_TEST_CASE ( script_CHECKMULTISIG12 )
{
2014-11-13 20:27:38 +01:00
ScriptError err ;
2011-08-27 00:41:22 +02:00
CKey key1 , key2 , key3 ;
2012-02-20 18:32:33 +01:00
key1 . MakeNewKey ( true ) ;
key2 . MakeNewKey ( false ) ;
key3 . MakeNewKey ( true ) ;
2011-08-27 00:41:22 +02:00
CScript scriptPubKey12 ;
2014-09-25 04:54:08 +02:00
scriptPubKey12 < < OP_1 < < ToByteVector ( key1 . GetPubKey ( ) ) < < ToByteVector ( key2 . GetPubKey ( ) ) < < OP_2 < < OP_CHECKMULTISIG ;
2011-08-27 00:41:22 +02:00
2014-09-28 06:33:33 +02:00
CMutableTransaction txFrom12 = BuildCreditingTransaction ( scriptPubKey12 ) ;
CMutableTransaction txTo12 = BuildSpendingTransaction ( CScript ( ) , txFrom12 ) ;
2011-08-27 00:41:22 +02:00
CScript goodsig1 = sign_multisig ( scriptPubKey12 , key1 , txTo12 ) ;
2015-01-27 15:01:31 +01:00
BOOST_CHECK ( VerifyScript ( goodsig1 , scriptPubKey12 , flags , MutableTransactionSignatureChecker ( & txTo12 , 0 ) , & err ) ) ;
2014-11-13 20:27:38 +01:00
BOOST_CHECK_MESSAGE ( err = = SCRIPT_ERR_OK , ScriptErrorString ( err ) ) ;
2011-08-27 00:41:22 +02:00
txTo12 . vout [ 0 ] . nValue = 2 ;
2015-01-27 15:01:31 +01:00
BOOST_CHECK ( ! VerifyScript ( goodsig1 , scriptPubKey12 , flags , MutableTransactionSignatureChecker ( & txTo12 , 0 ) , & err ) ) ;
2014-11-13 20:27:38 +01:00
BOOST_CHECK_MESSAGE ( err = = SCRIPT_ERR_EVAL_FALSE , ScriptErrorString ( err ) ) ;
2011-08-27 00:41:22 +02:00
CScript goodsig2 = sign_multisig ( scriptPubKey12 , key2 , txTo12 ) ;
2015-01-27 15:01:31 +01:00
BOOST_CHECK ( VerifyScript ( goodsig2 , scriptPubKey12 , flags , MutableTransactionSignatureChecker ( & txTo12 , 0 ) , & err ) ) ;
2014-11-13 20:27:38 +01:00
BOOST_CHECK_MESSAGE ( err = = SCRIPT_ERR_OK , ScriptErrorString ( err ) ) ;
2011-08-27 00:41:22 +02:00
CScript badsig1 = sign_multisig ( scriptPubKey12 , key3 , txTo12 ) ;
2015-01-27 15:01:31 +01:00
BOOST_CHECK ( ! VerifyScript ( badsig1 , scriptPubKey12 , flags , MutableTransactionSignatureChecker ( & txTo12 , 0 ) , & err ) ) ;
2014-11-13 20:27:38 +01:00
BOOST_CHECK_MESSAGE ( err = = SCRIPT_ERR_EVAL_FALSE , ScriptErrorString ( err ) ) ;
2011-08-27 00:41:22 +02:00
}
BOOST_AUTO_TEST_CASE ( script_CHECKMULTISIG23 )
{
2014-11-13 20:27:38 +01:00
ScriptError err ;
2011-08-27 00:41:22 +02:00
CKey key1 , key2 , key3 , key4 ;
2012-02-20 18:32:33 +01:00
key1 . MakeNewKey ( true ) ;
key2 . MakeNewKey ( false ) ;
key3 . MakeNewKey ( true ) ;
key4 . MakeNewKey ( false ) ;
2011-08-27 00:41:22 +02:00
CScript scriptPubKey23 ;
2014-09-25 04:54:08 +02:00
scriptPubKey23 < < OP_2 < < ToByteVector ( key1 . GetPubKey ( ) ) < < ToByteVector ( key2 . GetPubKey ( ) ) < < ToByteVector ( key3 . GetPubKey ( ) ) < < OP_3 < < OP_CHECKMULTISIG ;
2011-08-27 00:41:22 +02:00
2014-09-28 06:33:33 +02:00
CMutableTransaction txFrom23 = BuildCreditingTransaction ( scriptPubKey23 ) ;
CMutableTransaction txTo23 = BuildSpendingTransaction ( CScript ( ) , txFrom23 ) ;
2011-08-27 00:41:22 +02:00
std : : vector < CKey > keys ;
keys . push_back ( key1 ) ; keys . push_back ( key2 ) ;
CScript goodsig1 = sign_multisig ( scriptPubKey23 , keys , txTo23 ) ;
2015-01-27 15:01:31 +01:00
BOOST_CHECK ( VerifyScript ( goodsig1 , scriptPubKey23 , flags , MutableTransactionSignatureChecker ( & txTo23 , 0 ) , & err ) ) ;
2014-11-13 20:27:38 +01:00
BOOST_CHECK_MESSAGE ( err = = SCRIPT_ERR_OK , ScriptErrorString ( err ) ) ;
2011-08-27 00:41:22 +02:00
keys . clear ( ) ;
keys . push_back ( key1 ) ; keys . push_back ( key3 ) ;
CScript goodsig2 = sign_multisig ( scriptPubKey23 , keys , txTo23 ) ;
2015-01-27 15:01:31 +01:00
BOOST_CHECK ( VerifyScript ( goodsig2 , scriptPubKey23 , flags , MutableTransactionSignatureChecker ( & txTo23 , 0 ) , & err ) ) ;
2014-11-13 20:27:38 +01:00
BOOST_CHECK_MESSAGE ( err = = SCRIPT_ERR_OK , ScriptErrorString ( err ) ) ;
2011-08-27 00:41:22 +02:00
keys . clear ( ) ;
keys . push_back ( key2 ) ; keys . push_back ( key3 ) ;
CScript goodsig3 = sign_multisig ( scriptPubKey23 , keys , txTo23 ) ;
2015-01-27 15:01:31 +01:00
BOOST_CHECK ( VerifyScript ( goodsig3 , scriptPubKey23 , flags , MutableTransactionSignatureChecker ( & txTo23 , 0 ) , & err ) ) ;
2014-11-13 20:27:38 +01:00
BOOST_CHECK_MESSAGE ( err = = SCRIPT_ERR_OK , ScriptErrorString ( err ) ) ;
2011-08-27 00:41:22 +02:00
keys . clear ( ) ;
keys . push_back ( key2 ) ; keys . push_back ( key2 ) ; // Can't re-use sig
CScript badsig1 = sign_multisig ( scriptPubKey23 , keys , txTo23 ) ;
2015-01-27 15:01:31 +01:00
BOOST_CHECK ( ! VerifyScript ( badsig1 , scriptPubKey23 , flags , MutableTransactionSignatureChecker ( & txTo23 , 0 ) , & err ) ) ;
2014-11-13 20:27:38 +01:00
BOOST_CHECK_MESSAGE ( err = = SCRIPT_ERR_EVAL_FALSE , ScriptErrorString ( err ) ) ;
2011-08-27 00:41:22 +02:00
keys . clear ( ) ;
keys . push_back ( key2 ) ; keys . push_back ( key1 ) ; // sigs must be in correct order
CScript badsig2 = sign_multisig ( scriptPubKey23 , keys , txTo23 ) ;
2015-01-27 15:01:31 +01:00
BOOST_CHECK ( ! VerifyScript ( badsig2 , scriptPubKey23 , flags , MutableTransactionSignatureChecker ( & txTo23 , 0 ) , & err ) ) ;
2014-11-13 20:27:38 +01:00
BOOST_CHECK_MESSAGE ( err = = SCRIPT_ERR_EVAL_FALSE , ScriptErrorString ( err ) ) ;
2011-08-27 00:41:22 +02:00
keys . clear ( ) ;
keys . push_back ( key3 ) ; keys . push_back ( key2 ) ; // sigs must be in correct order
CScript badsig3 = sign_multisig ( scriptPubKey23 , keys , txTo23 ) ;
2015-01-27 15:01:31 +01:00
BOOST_CHECK ( ! VerifyScript ( badsig3 , scriptPubKey23 , flags , MutableTransactionSignatureChecker ( & txTo23 , 0 ) , & err ) ) ;
2014-11-13 20:27:38 +01:00
BOOST_CHECK_MESSAGE ( err = = SCRIPT_ERR_EVAL_FALSE , ScriptErrorString ( err ) ) ;
2011-08-27 00:41:22 +02:00
keys . clear ( ) ;
keys . push_back ( key4 ) ; keys . push_back ( key2 ) ; // sigs must match pubkeys
CScript badsig4 = sign_multisig ( scriptPubKey23 , keys , txTo23 ) ;
2015-01-27 15:01:31 +01:00
BOOST_CHECK ( ! VerifyScript ( badsig4 , scriptPubKey23 , flags , MutableTransactionSignatureChecker ( & txTo23 , 0 ) , & err ) ) ;
2014-11-13 20:27:38 +01:00
BOOST_CHECK_MESSAGE ( err = = SCRIPT_ERR_EVAL_FALSE , ScriptErrorString ( err ) ) ;
2011-08-27 00:41:22 +02:00
keys . clear ( ) ;
keys . push_back ( key1 ) ; keys . push_back ( key4 ) ; // sigs must match pubkeys
CScript badsig5 = sign_multisig ( scriptPubKey23 , keys , txTo23 ) ;
2015-01-27 15:01:31 +01:00
BOOST_CHECK ( ! VerifyScript ( badsig5 , scriptPubKey23 , flags , MutableTransactionSignatureChecker ( & txTo23 , 0 ) , & err ) ) ;
2014-11-13 20:27:38 +01:00
BOOST_CHECK_MESSAGE ( err = = SCRIPT_ERR_EVAL_FALSE , ScriptErrorString ( err ) ) ;
2011-08-27 00:41:22 +02:00
keys . clear ( ) ; // Must have signatures
CScript badsig6 = sign_multisig ( scriptPubKey23 , keys , txTo23 ) ;
2015-01-27 15:01:31 +01:00
BOOST_CHECK ( ! VerifyScript ( badsig6 , scriptPubKey23 , flags , MutableTransactionSignatureChecker ( & txTo23 , 0 ) , & err ) ) ;
2014-11-13 20:27:38 +01:00
BOOST_CHECK_MESSAGE ( err = = SCRIPT_ERR_INVALID_STACK_OPERATION , ScriptErrorString ( err ) ) ;
2011-08-27 00:41:22 +02:00
}
2012-05-31 22:01:16 +02:00
BOOST_AUTO_TEST_CASE ( script_combineSigs )
{
// Test the CombineSignatures function
CBasicKeyStore keystore ;
vector < CKey > keys ;
2013-05-01 06:52:05 +02:00
vector < CPubKey > pubkeys ;
2012-05-31 22:01:16 +02:00
for ( int i = 0 ; i < 3 ; i + + )
{
CKey key ;
key . MakeNewKey ( i % 2 = = 1 ) ;
keys . push_back ( key ) ;
2013-05-01 06:52:05 +02:00
pubkeys . push_back ( key . GetPubKey ( ) ) ;
2012-05-31 22:01:16 +02:00
keystore . AddKey ( key ) ;
}
2014-09-28 06:33:33 +02:00
CMutableTransaction txFrom = BuildCreditingTransaction ( GetScriptForDestination ( keys [ 0 ] . GetPubKey ( ) . GetID ( ) ) ) ;
CMutableTransaction txTo = BuildSpendingTransaction ( CScript ( ) , txFrom ) ;
2012-05-31 22:01:16 +02:00
CScript & scriptPubKey = txFrom . vout [ 0 ] . scriptPubKey ;
CScript & scriptSig = txTo . vin [ 0 ] . scriptSig ;
CScript empty ;
CScript combined = CombineSignatures ( scriptPubKey , txTo , 0 , empty , empty ) ;
BOOST_CHECK ( combined . empty ( ) ) ;
// Single signature case:
SignSignature ( keystore , txFrom , txTo , 0 ) ; // changes scriptSig
combined = CombineSignatures ( scriptPubKey , txTo , 0 , scriptSig , empty ) ;
BOOST_CHECK ( combined = = scriptSig ) ;
combined = CombineSignatures ( scriptPubKey , txTo , 0 , empty , scriptSig ) ;
BOOST_CHECK ( combined = = scriptSig ) ;
CScript scriptSigCopy = scriptSig ;
// Signing again will give a different, valid signature:
SignSignature ( keystore , txFrom , txTo , 0 ) ;
combined = CombineSignatures ( scriptPubKey , txTo , 0 , scriptSigCopy , scriptSig ) ;
BOOST_CHECK ( combined = = scriptSigCopy | | combined = = scriptSig ) ;
// P2SH, single-signature case:
2014-09-25 04:54:08 +02:00
CScript pkSingle ; pkSingle < < ToByteVector ( keys [ 0 ] . GetPubKey ( ) ) < < OP_CHECKSIG ;
2012-05-31 22:01:16 +02:00
keystore . AddCScript ( pkSingle ) ;
2014-09-25 04:24:46 +02:00
scriptPubKey = GetScriptForDestination ( CScriptID ( pkSingle ) ) ;
2012-05-31 22:01:16 +02:00
SignSignature ( keystore , txFrom , txTo , 0 ) ;
combined = CombineSignatures ( scriptPubKey , txTo , 0 , scriptSig , empty ) ;
BOOST_CHECK ( combined = = scriptSig ) ;
combined = CombineSignatures ( scriptPubKey , txTo , 0 , empty , scriptSig ) ;
BOOST_CHECK ( combined = = scriptSig ) ;
scriptSigCopy = scriptSig ;
SignSignature ( keystore , txFrom , txTo , 0 ) ;
combined = CombineSignatures ( scriptPubKey , txTo , 0 , scriptSigCopy , scriptSig ) ;
BOOST_CHECK ( combined = = scriptSigCopy | | combined = = scriptSig ) ;
// dummy scriptSigCopy with placeholder, should always choose non-placeholder:
scriptSigCopy = CScript ( ) < < OP_0 < < static_cast < vector < unsigned char > > ( pkSingle ) ;
combined = CombineSignatures ( scriptPubKey , txTo , 0 , scriptSigCopy , scriptSig ) ;
BOOST_CHECK ( combined = = scriptSig ) ;
combined = CombineSignatures ( scriptPubKey , txTo , 0 , scriptSig , scriptSigCopy ) ;
BOOST_CHECK ( combined = = scriptSig ) ;
// Hardest case: Multisig 2-of-3
2014-09-11 19:15:29 +02:00
scriptPubKey = GetScriptForMultisig ( 2 , pubkeys ) ;
2012-05-31 22:01:16 +02:00
keystore . AddCScript ( scriptPubKey ) ;
SignSignature ( keystore , txFrom , txTo , 0 ) ;
combined = CombineSignatures ( scriptPubKey , txTo , 0 , scriptSig , empty ) ;
BOOST_CHECK ( combined = = scriptSig ) ;
combined = CombineSignatures ( scriptPubKey , txTo , 0 , empty , scriptSig ) ;
BOOST_CHECK ( combined = = scriptSig ) ;
// A couple of partially-signed versions:
vector < unsigned char > sig1 ;
uint256 hash1 = SignatureHash ( scriptPubKey , txTo , 0 , SIGHASH_ALL ) ;
BOOST_CHECK ( keys [ 0 ] . Sign ( hash1 , sig1 ) ) ;
sig1 . push_back ( SIGHASH_ALL ) ;
vector < unsigned char > sig2 ;
uint256 hash2 = SignatureHash ( scriptPubKey , txTo , 0 , SIGHASH_NONE ) ;
BOOST_CHECK ( keys [ 1 ] . Sign ( hash2 , sig2 ) ) ;
sig2 . push_back ( SIGHASH_NONE ) ;
vector < unsigned char > sig3 ;
uint256 hash3 = SignatureHash ( scriptPubKey , txTo , 0 , SIGHASH_SINGLE ) ;
BOOST_CHECK ( keys [ 2 ] . Sign ( hash3 , sig3 ) ) ;
sig3 . push_back ( SIGHASH_SINGLE ) ;
// Not fussy about order (or even existence) of placeholders or signatures:
CScript partial1a = CScript ( ) < < OP_0 < < sig1 < < OP_0 ;
CScript partial1b = CScript ( ) < < OP_0 < < OP_0 < < sig1 ;
CScript partial2a = CScript ( ) < < OP_0 < < sig2 ;
CScript partial2b = CScript ( ) < < sig2 < < OP_0 ;
CScript partial3a = CScript ( ) < < sig3 ;
CScript partial3b = CScript ( ) < < OP_0 < < OP_0 < < sig3 ;
CScript partial3c = CScript ( ) < < OP_0 < < sig3 < < OP_0 ;
CScript complete12 = CScript ( ) < < OP_0 < < sig1 < < sig2 ;
CScript complete13 = CScript ( ) < < OP_0 < < sig1 < < sig3 ;
CScript complete23 = CScript ( ) < < OP_0 < < sig2 < < sig3 ;
combined = CombineSignatures ( scriptPubKey , txTo , 0 , partial1a , partial1b ) ;
BOOST_CHECK ( combined = = partial1a ) ;
combined = CombineSignatures ( scriptPubKey , txTo , 0 , partial1a , partial2a ) ;
BOOST_CHECK ( combined = = complete12 ) ;
combined = CombineSignatures ( scriptPubKey , txTo , 0 , partial2a , partial1a ) ;
BOOST_CHECK ( combined = = complete12 ) ;
combined = CombineSignatures ( scriptPubKey , txTo , 0 , partial1b , partial2b ) ;
BOOST_CHECK ( combined = = complete12 ) ;
combined = CombineSignatures ( scriptPubKey , txTo , 0 , partial3b , partial1b ) ;
BOOST_CHECK ( combined = = complete13 ) ;
combined = CombineSignatures ( scriptPubKey , txTo , 0 , partial2a , partial3a ) ;
BOOST_CHECK ( combined = = complete23 ) ;
combined = CombineSignatures ( scriptPubKey , txTo , 0 , partial3b , partial2b ) ;
BOOST_CHECK ( combined = = complete23 ) ;
combined = CombineSignatures ( scriptPubKey , txTo , 0 , partial3b , partial3a ) ;
BOOST_CHECK ( combined = = partial3c ) ;
}
2011-08-27 00:41:22 +02:00
2013-09-24 00:48:00 +02:00
BOOST_AUTO_TEST_CASE ( script_standard_push )
{
2014-11-13 20:27:38 +01:00
ScriptError err ;
2014-10-09 03:48:59 +02:00
for ( int i = 0 ; i < 67000 ; i + + ) {
2013-09-24 00:48:00 +02:00
CScript script ;
script < < i ;
BOOST_CHECK_MESSAGE ( script . IsPushOnly ( ) , " Number " < < i < < " is not pure push. " ) ;
2014-11-13 20:27:38 +01:00
BOOST_CHECK_MESSAGE ( VerifyScript ( script , CScript ( ) < < OP_1 , SCRIPT_VERIFY_MINIMALDATA , BaseSignatureChecker ( ) , & err ) , " Number " < < i < < " push is not minimal data. " ) ;
BOOST_CHECK_MESSAGE ( err = = SCRIPT_ERR_OK , ScriptErrorString ( err ) ) ;
2013-09-24 00:48:00 +02:00
}
2014-10-09 03:48:59 +02:00
for ( unsigned int i = 0 ; i < = MAX_SCRIPT_ELEMENT_SIZE ; i + + ) {
2013-09-24 00:48:00 +02:00
std : : vector < unsigned char > data ( i , ' \111 ' ) ;
CScript script ;
script < < data ;
BOOST_CHECK_MESSAGE ( script . IsPushOnly ( ) , " Length " < < i < < " is not pure push. " ) ;
2014-11-13 20:27:38 +01:00
BOOST_CHECK_MESSAGE ( VerifyScript ( script , CScript ( ) < < OP_1 , SCRIPT_VERIFY_MINIMALDATA , BaseSignatureChecker ( ) , & err ) , " Length " < < i < < " push is not minimal data. " ) ;
BOOST_CHECK_MESSAGE ( err = = SCRIPT_ERR_OK , ScriptErrorString ( err ) ) ;
2013-09-24 00:48:00 +02:00
}
}
2014-08-25 03:16:51 +02:00
BOOST_AUTO_TEST_CASE ( script_IsPushOnly_on_invalid_scripts )
{
// IsPushOnly returns false when given a script containing only pushes that
// are invalid due to truncation. IsPushOnly() is consensus critical
// because P2SH evaluation uses it, although this specific behavior should
// not be consensus critical as the P2SH evaluation would fail first due to
// the invalid push. Still, it doesn't hurt to test it explicitly.
static const unsigned char direct [ ] = { 1 } ;
BOOST_CHECK ( ! CScript ( direct , direct + sizeof ( direct ) ) . IsPushOnly ( ) ) ;
}
2011-07-31 20:01:31 +02:00
BOOST_AUTO_TEST_SUITE_END ( )