2014-08-27 17:22:33 +02:00
// Copyright (c) 2009-2010 Satoshi Nakamoto
2018-07-26 18:36:45 -04:00
// Copyright (c) 2009-2018 The Bitcoin Core developers
2014-09-09 10:00:42 +02:00
// Distributed under the MIT software license, see the accompanying
2014-08-27 17:22:33 +02:00
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
2014-11-03 16:16:40 +01:00
# ifndef BITCOIN_SCRIPT_SIGN_H
# define BITCOIN_SCRIPT_SIGN_H
2014-08-27 17:22:33 +02:00
2018-06-27 16:53:48 -07:00
# include <boost/optional.hpp>
# include <hash.h>
# include <pubkey.h>
2017-11-10 13:57:53 +13:00
# include <script/interpreter.h>
2018-06-27 16:53:48 -07:00
# include <streams.h>
2014-08-27 17:22:33 +02:00
2018-03-17 19:19:09 -07:00
class CKey ;
2014-11-04 10:06:20 -08:00
class CKeyID ;
2014-08-27 17:22:33 +02:00
class CScript ;
2018-03-17 19:19:09 -07:00
class CScriptID ;
2014-08-27 17:22:33 +02:00
class CTransaction ;
2014-09-09 10:00:42 +02:00
2014-08-27 17:22:33 +02:00
struct CMutableTransaction ;
2018-07-19 23:15:53 -07:00
struct KeyOriginInfo
{
unsigned char fingerprint [ 4 ] ;
std : : vector < uint32_t > path ;
} ;
2018-03-17 19:19:09 -07:00
/** An interface to be implemented by keystores that support signing. */
class SigningProvider
{
public :
virtual ~ SigningProvider ( ) { }
2018-06-07 21:12:25 -07:00
virtual bool GetCScript ( const CScriptID & scriptid , CScript & script ) const { return false ; }
virtual bool GetPubKey ( const CKeyID & address , CPubKey & pubkey ) const { return false ; }
virtual bool GetKey ( const CKeyID & address , CKey & key ) const { return false ; }
2018-03-17 19:19:09 -07:00
} ;
2018-06-07 21:12:25 -07:00
extern const SigningProvider & DUMMY_SIGNING_PROVIDER ;
2018-06-28 19:05:05 -07:00
class PublicOnlySigningProvider : public SigningProvider
{
private :
const SigningProvider * m_provider ;
public :
PublicOnlySigningProvider ( const SigningProvider * provider ) : m_provider ( provider ) { }
bool GetCScript ( const CScriptID & scriptid , CScript & script ) const ;
bool GetPubKey ( const CKeyID & address , CPubKey & pubkey ) const ;
} ;
2018-07-12 15:03:55 -07:00
struct FlatSigningProvider final : public SigningProvider
{
std : : map < CScriptID , CScript > scripts ;
std : : map < CKeyID , CPubKey > pubkeys ;
std : : map < CKeyID , CKey > keys ;
bool GetCScript ( const CScriptID & scriptid , CScript & script ) const override ;
bool GetPubKey ( const CKeyID & keyid , CPubKey & pubkey ) const override ;
bool GetKey ( const CKeyID & keyid , CKey & key ) const override ;
} ;
FlatSigningProvider Merge ( const FlatSigningProvider & a , const FlatSigningProvider & b ) ;
2018-03-27 13:15:10 -07:00
/** Interface for signature creators. */
2014-11-04 10:06:20 -08:00
class BaseSignatureCreator {
public :
virtual ~ BaseSignatureCreator ( ) { }
virtual const BaseSignatureChecker & Checker ( ) const = 0 ;
/** Create a singular (non-script) signature. */
2018-03-27 13:15:10 -07:00
virtual bool CreateSig ( const SigningProvider & provider , std : : vector < unsigned char > & vchSig , const CKeyID & keyid , const CScript & scriptCode , SigVersion sigversion ) const = 0 ;
2014-11-04 10:06:20 -08:00
} ;
/** A signature creator for transactions. */
2018-05-20 22:47:14 +02:00
class MutableTransactionSignatureCreator : public BaseSignatureCreator {
const CMutableTransaction * txTo ;
2014-11-04 10:06:20 -08:00
unsigned int nIn ;
int nHashType ;
2015-12-27 19:49:08 +01:00
CAmount amount ;
2018-05-20 22:47:14 +02:00
const MutableTransactionSignatureChecker checker ;
2014-11-04 10:06:20 -08:00
public :
2018-05-20 22:47:14 +02:00
MutableTransactionSignatureCreator ( const CMutableTransaction * txToIn , unsigned int nInIn , const CAmount & amountIn , int nHashTypeIn = SIGHASH_ALL ) ;
2017-06-20 21:58:56 +02:00
const BaseSignatureChecker & Checker ( ) const override { return checker ; }
2018-03-27 13:15:10 -07:00
bool CreateSig ( const SigningProvider & provider , std : : vector < unsigned char > & vchSig , const CKeyID & keyid , const CScript & scriptCode , SigVersion sigversion ) const override ;
2016-03-31 14:54:58 +02:00
} ;
2018-08-07 16:59:53 -07:00
/** A signature creator that just produces 71-byte empty signatures. */
2018-03-27 13:34:39 -07:00
extern const BaseSignatureCreator & DUMMY_SIGNATURE_CREATOR ;
2018-08-07 16:59:53 -07:00
/** A signature creator that just produces 72-byte empty signatures. */
extern const BaseSignatureCreator & DUMMY_MAXIMUM_SIGNATURE_CREATOR ;
2016-03-31 14:54:58 +02:00
2018-07-03 17:18:52 -07:00
typedef std : : pair < CPubKey , std : : vector < unsigned char > > SigPair ;
// This struct contains information from a transaction input and also contains signatures for that input.
// The information contained here can be used to create a signature and is also filled by ProduceSignature
// in order to construct final scriptSigs and scriptWitnesses.
2016-03-31 14:54:58 +02:00
struct SignatureData {
2018-07-03 17:18:52 -07:00
bool complete = false ; ///< Stores whether the scriptSig and scriptWitness are complete
2018-06-27 16:56:30 -07:00
bool witness = false ; ///< Stores whether the input this SigData corresponds to is a witness input
2018-07-03 17:18:52 -07:00
CScript scriptSig ; ///< The scriptSig of an input. Contains complete signatures or the traditional partial signatures format
CScript redeem_script ; ///< The redeemScript (if any) for the input
CScript witness_script ; ///< The witnessScript (if any) for the input. witnessScripts are used in P2WSH outputs.
CScriptWitness scriptWitness ; ///< The scriptWitness of an input. Contains complete signatures or the traditional partial signatures format. scriptWitness is part of a transaction input per BIP 144.
std : : map < CKeyID , SigPair > signatures ; ///< BIP 174 style partial signatures for the input. May contain all signatures necessary for producing a final scriptSig or scriptWitness.
2018-06-27 16:56:30 -07:00
std : : map < CKeyID , CPubKey > misc_pubkeys ;
2016-03-31 14:54:58 +02:00
SignatureData ( ) { }
explicit SignatureData ( const CScript & script ) : scriptSig ( script ) { }
2018-06-07 21:12:25 -07:00
void MergeSignatureData ( SignatureData sigdata ) ;
2015-05-01 15:21:06 +02:00
} ;
2018-06-27 16:53:48 -07:00
// Magic bytes
static constexpr uint8_t PSBT_MAGIC_BYTES [ 5 ] = { ' p ' , ' s ' , ' b ' , ' t ' , 0xff } ;
// Global types
static constexpr uint8_t PSBT_GLOBAL_UNSIGNED_TX = 0x00 ;
// Input types
static constexpr uint8_t PSBT_IN_NON_WITNESS_UTXO = 0x00 ;
static constexpr uint8_t PSBT_IN_WITNESS_UTXO = 0x01 ;
static constexpr uint8_t PSBT_IN_PARTIAL_SIG = 0x02 ;
static constexpr uint8_t PSBT_IN_SIGHASH = 0x03 ;
static constexpr uint8_t PSBT_IN_REDEEMSCRIPT = 0x04 ;
static constexpr uint8_t PSBT_IN_WITNESSSCRIPT = 0x05 ;
static constexpr uint8_t PSBT_IN_BIP32_DERIVATION = 0x06 ;
static constexpr uint8_t PSBT_IN_SCRIPTSIG = 0x07 ;
static constexpr uint8_t PSBT_IN_SCRIPTWITNESS = 0x08 ;
// Output types
static constexpr uint8_t PSBT_OUT_REDEEMSCRIPT = 0x00 ;
static constexpr uint8_t PSBT_OUT_WITNESSSCRIPT = 0x01 ;
static constexpr uint8_t PSBT_OUT_BIP32_DERIVATION = 0x02 ;
// The separator is 0x00. Reading this in means that the unserializer can interpret it
// as a 0 length key which indicates that this is the separator. The separator has no value.
static constexpr uint8_t PSBT_SEPARATOR = 0x00 ;
2018-07-18 17:52:43 -07:00
// Takes a stream and multiple arguments and serializes them as if first serialized into a vector and then into the stream
2018-06-27 16:53:48 -07:00
// The resulting output into the stream has the total serialized length of all of the objects followed by all objects concatenated with each other.
template < typename Stream , typename . . . X >
void SerializeToVector ( Stream & s , const X & . . . args )
{
2018-07-18 17:52:43 -07:00
WriteCompactSize ( s , GetSerializeSizeMany ( s , args . . . ) ) ;
SerializeMany ( s , args . . . ) ;
2018-06-27 16:53:48 -07:00
}
// Takes a stream and multiple arguments and unserializes them first as a vector then each object individually in the order provided in the arguments
template < typename Stream , typename . . . X >
void UnserializeFromVector ( Stream & s , X & . . . args )
{
2018-07-18 17:52:43 -07:00
size_t expected_size = ReadCompactSize ( s ) ;
size_t remaining_before = s . size ( ) ;
UnserializeMany ( s , args . . . ) ;
size_t remaining_after = s . size ( ) ;
if ( remaining_after + expected_size ! = remaining_before ) {
2018-06-27 16:53:48 -07:00
throw std : : ios_base : : failure ( " Size of value was not the stated size " ) ;
}
}
// Deserialize HD keypaths into a map
template < typename Stream >
2018-07-19 23:15:53 -07:00
void DeserializeHDKeypaths ( Stream & s , const std : : vector < unsigned char > & key , std : : map < CPubKey , KeyOriginInfo > & hd_keypaths )
2018-06-27 16:53:48 -07:00
{
// Make sure that the key is the size of pubkey + 1
if ( key . size ( ) ! = CPubKey : : PUBLIC_KEY_SIZE + 1 & & key . size ( ) ! = CPubKey : : COMPRESSED_PUBLIC_KEY_SIZE + 1 ) {
throw std : : ios_base : : failure ( " Size of key was not the expected size for the type BIP32 keypath " ) ;
}
// Read in the pubkey from key
CPubKey pubkey ( key . begin ( ) + 1 , key . end ( ) ) ;
if ( ! pubkey . IsFullyValid ( ) ) {
throw std : : ios_base : : failure ( " Invalid pubkey " ) ;
}
if ( hd_keypaths . count ( pubkey ) > 0 ) {
throw std : : ios_base : : failure ( " Duplicate Key, pubkey derivation path already provided " ) ;
}
// Read in key path
uint64_t value_len = ReadCompactSize ( s ) ;
2018-07-19 23:15:53 -07:00
if ( value_len % 4 | | value_len = = 0 ) {
throw std : : ios_base : : failure ( " Invalid length for HD key path " ) ;
}
KeyOriginInfo keypath ;
s > > keypath . fingerprint ;
for ( unsigned int i = 4 ; i < value_len ; i + = sizeof ( uint32_t ) ) {
2018-06-27 16:53:48 -07:00
uint32_t index ;
s > > index ;
2018-07-19 23:15:53 -07:00
keypath . path . push_back ( index ) ;
2018-06-27 16:53:48 -07:00
}
// Add to map
2018-07-19 23:15:53 -07:00
hd_keypaths . emplace ( pubkey , std : : move ( keypath ) ) ;
2018-06-27 16:53:48 -07:00
}
// Serialize HD keypaths to a stream from a map
template < typename Stream >
2018-07-19 23:15:53 -07:00
void SerializeHDKeypaths ( Stream & s , const std : : map < CPubKey , KeyOriginInfo > & hd_keypaths , uint8_t type )
2018-06-27 16:53:48 -07:00
{
for ( auto keypath_pair : hd_keypaths ) {
SerializeToVector ( s , type , MakeSpan ( keypath_pair . first ) ) ;
2018-07-19 23:15:53 -07:00
WriteCompactSize ( s , ( keypath_pair . second . path . size ( ) + 1 ) * sizeof ( uint32_t ) ) ;
s < < keypath_pair . second . fingerprint ;
for ( const auto & path : keypath_pair . second . path ) {
2018-06-27 16:53:48 -07:00
s < < path ;
}
}
}
/** A structure for PSBTs which contain per-input information */
struct PSBTInput
{
CTransactionRef non_witness_utxo ;
CTxOut witness_utxo ;
CScript redeem_script ;
CScript witness_script ;
CScript final_script_sig ;
CScriptWitness final_script_witness ;
2018-07-19 23:15:53 -07:00
std : : map < CPubKey , KeyOriginInfo > hd_keypaths ;
2018-06-27 16:53:48 -07:00
std : : map < CKeyID , SigPair > partial_sigs ;
std : : map < std : : vector < unsigned char > , std : : vector < unsigned char > > unknown ;
int sighash_type = 0 ;
bool IsNull ( ) const ;
2018-06-27 16:58:01 -07:00
void FillSignatureData ( SignatureData & sigdata ) const ;
void FromSignatureData ( const SignatureData & sigdata ) ;
void Merge ( const PSBTInput & input ) ;
bool IsSane ( ) const ;
2018-06-27 16:53:48 -07:00
PSBTInput ( ) { }
template < typename Stream >
inline void Serialize ( Stream & s ) const {
// Write the utxo
// If there is a non-witness utxo, then don't add the witness one.
if ( non_witness_utxo ) {
SerializeToVector ( s , PSBT_IN_NON_WITNESS_UTXO ) ;
SerializeToVector ( s , non_witness_utxo ) ;
} else if ( ! witness_utxo . IsNull ( ) ) {
SerializeToVector ( s , PSBT_IN_WITNESS_UTXO ) ;
SerializeToVector ( s , witness_utxo ) ;
}
if ( final_script_sig . empty ( ) & & final_script_witness . IsNull ( ) ) {
// Write any partial signatures
for ( auto sig_pair : partial_sigs ) {
SerializeToVector ( s , PSBT_IN_PARTIAL_SIG , MakeSpan ( sig_pair . second . first ) ) ;
s < < sig_pair . second . second ;
}
// Write the sighash type
if ( sighash_type > 0 ) {
SerializeToVector ( s , PSBT_IN_SIGHASH ) ;
SerializeToVector ( s , sighash_type ) ;
}
// Write the redeem script
if ( ! redeem_script . empty ( ) ) {
SerializeToVector ( s , PSBT_IN_REDEEMSCRIPT ) ;
s < < redeem_script ;
}
// Write the witness script
if ( ! witness_script . empty ( ) ) {
SerializeToVector ( s , PSBT_IN_WITNESSSCRIPT ) ;
s < < witness_script ;
}
// Write any hd keypaths
SerializeHDKeypaths ( s , hd_keypaths , PSBT_IN_BIP32_DERIVATION ) ;
}
// Write script sig
if ( ! final_script_sig . empty ( ) ) {
SerializeToVector ( s , PSBT_IN_SCRIPTSIG ) ;
s < < final_script_sig ;
}
// write script witness
if ( ! final_script_witness . IsNull ( ) ) {
SerializeToVector ( s , PSBT_IN_SCRIPTWITNESS ) ;
SerializeToVector ( s , final_script_witness . stack ) ;
}
// Write unknown things
for ( auto & entry : unknown ) {
s < < entry . first ;
s < < entry . second ;
}
s < < PSBT_SEPARATOR ;
}
template < typename Stream >
inline void Unserialize ( Stream & s ) {
// Read loop
while ( ! s . empty ( ) ) {
// Read
std : : vector < unsigned char > key ;
s > > key ;
// the key is empty if that was actually a separator byte
// This is a special case for key lengths 0 as those are not allowed (except for separator)
if ( key . empty ( ) ) return ;
// First byte of key is the type
unsigned char type = key [ 0 ] ;
// Do stuff based on type
switch ( type ) {
case PSBT_IN_NON_WITNESS_UTXO :
if ( non_witness_utxo ) {
throw std : : ios_base : : failure ( " Duplicate Key, input non-witness utxo already provided " ) ;
2018-07-19 18:29:25 -07:00
} else if ( key . size ( ) ! = 1 ) {
throw std : : ios_base : : failure ( " Non-witness utxo key is more than one byte type " ) ;
2018-06-27 16:53:48 -07:00
}
UnserializeFromVector ( s , non_witness_utxo ) ;
break ;
case PSBT_IN_WITNESS_UTXO :
if ( ! witness_utxo . IsNull ( ) ) {
throw std : : ios_base : : failure ( " Duplicate Key, input witness utxo already provided " ) ;
2018-07-19 18:29:25 -07:00
} else if ( key . size ( ) ! = 1 ) {
throw std : : ios_base : : failure ( " Witness utxo key is more than one byte type " ) ;
2018-06-27 16:53:48 -07:00
}
UnserializeFromVector ( s , witness_utxo ) ;
break ;
case PSBT_IN_PARTIAL_SIG :
{
// Make sure that the key is the size of pubkey + 1
if ( key . size ( ) ! = CPubKey : : PUBLIC_KEY_SIZE + 1 & & key . size ( ) ! = CPubKey : : COMPRESSED_PUBLIC_KEY_SIZE + 1 ) {
throw std : : ios_base : : failure ( " Size of key was not the expected size for the type partial signature pubkey " ) ;
}
// Read in the pubkey from key
CPubKey pubkey ( key . begin ( ) + 1 , key . end ( ) ) ;
if ( ! pubkey . IsFullyValid ( ) ) {
throw std : : ios_base : : failure ( " Invalid pubkey " ) ;
}
if ( partial_sigs . count ( pubkey . GetID ( ) ) > 0 ) {
throw std : : ios_base : : failure ( " Duplicate Key, input partial signature for pubkey already provided " ) ;
}
// Read in the signature from value
std : : vector < unsigned char > sig ;
s > > sig ;
// Add to list
partial_sigs . emplace ( pubkey . GetID ( ) , SigPair ( pubkey , std : : move ( sig ) ) ) ;
break ;
}
case PSBT_IN_SIGHASH :
if ( sighash_type > 0 ) {
throw std : : ios_base : : failure ( " Duplicate Key, input sighash type already provided " ) ;
2018-07-19 18:29:25 -07:00
} else if ( key . size ( ) ! = 1 ) {
throw std : : ios_base : : failure ( " Sighash type key is more than one byte type " ) ;
2018-06-27 16:53:48 -07:00
}
UnserializeFromVector ( s , sighash_type ) ;
break ;
case PSBT_IN_REDEEMSCRIPT :
{
if ( ! redeem_script . empty ( ) ) {
throw std : : ios_base : : failure ( " Duplicate Key, input redeemScript already provided " ) ;
2018-07-19 18:29:25 -07:00
} else if ( key . size ( ) ! = 1 ) {
throw std : : ios_base : : failure ( " Input redeemScript key is more than one byte type " ) ;
2018-06-27 16:53:48 -07:00
}
s > > redeem_script ;
break ;
}
case PSBT_IN_WITNESSSCRIPT :
{
if ( ! witness_script . empty ( ) ) {
throw std : : ios_base : : failure ( " Duplicate Key, input witnessScript already provided " ) ;
2018-07-19 18:29:25 -07:00
} else if ( key . size ( ) ! = 1 ) {
throw std : : ios_base : : failure ( " Input witnessScript key is more than one byte type " ) ;
2018-06-27 16:53:48 -07:00
}
s > > witness_script ;
break ;
}
case PSBT_IN_BIP32_DERIVATION :
{
DeserializeHDKeypaths ( s , key , hd_keypaths ) ;
break ;
}
case PSBT_IN_SCRIPTSIG :
{
if ( ! final_script_sig . empty ( ) ) {
throw std : : ios_base : : failure ( " Duplicate Key, input final scriptSig already provided " ) ;
2018-07-19 18:29:25 -07:00
} else if ( key . size ( ) ! = 1 ) {
throw std : : ios_base : : failure ( " Final scriptSig key is more than one byte type " ) ;
2018-06-27 16:53:48 -07:00
}
s > > final_script_sig ;
break ;
}
case PSBT_IN_SCRIPTWITNESS :
{
if ( ! final_script_witness . IsNull ( ) ) {
throw std : : ios_base : : failure ( " Duplicate Key, input final scriptWitness already provided " ) ;
2018-07-19 18:29:25 -07:00
} else if ( key . size ( ) ! = 1 ) {
throw std : : ios_base : : failure ( " Final scriptWitness key is more than one byte type " ) ;
2018-06-27 16:53:48 -07:00
}
UnserializeFromVector ( s , final_script_witness . stack ) ;
break ;
}
// Unknown stuff
default :
if ( unknown . count ( key ) > 0 ) {
throw std : : ios_base : : failure ( " Duplicate Key, key for unknown value already provided " ) ;
}
// Read in the value
std : : vector < unsigned char > val_bytes ;
s > > val_bytes ;
unknown . emplace ( std : : move ( key ) , std : : move ( val_bytes ) ) ;
break ;
}
}
}
template < typename Stream >
PSBTInput ( deserialize_type , Stream & s ) {
Unserialize ( s ) ;
}
} ;
/** A structure for PSBTs which contains per output information */
struct PSBTOutput
{
CScript redeem_script ;
CScript witness_script ;
2018-07-19 23:15:53 -07:00
std : : map < CPubKey , KeyOriginInfo > hd_keypaths ;
2018-06-27 16:53:48 -07:00
std : : map < std : : vector < unsigned char > , std : : vector < unsigned char > > unknown ;
bool IsNull ( ) const ;
2018-06-27 16:58:01 -07:00
void FillSignatureData ( SignatureData & sigdata ) const ;
void FromSignatureData ( const SignatureData & sigdata ) ;
void Merge ( const PSBTOutput & output ) ;
bool IsSane ( ) const ;
2018-06-27 16:53:48 -07:00
PSBTOutput ( ) { }
template < typename Stream >
inline void Serialize ( Stream & s ) const {
// Write the redeem script
if ( ! redeem_script . empty ( ) ) {
SerializeToVector ( s , PSBT_OUT_REDEEMSCRIPT ) ;
s < < redeem_script ;
}
// Write the witness script
if ( ! witness_script . empty ( ) ) {
SerializeToVector ( s , PSBT_OUT_WITNESSSCRIPT ) ;
s < < witness_script ;
}
// Write any hd keypaths
SerializeHDKeypaths ( s , hd_keypaths , PSBT_OUT_BIP32_DERIVATION ) ;
// Write unknown things
for ( auto & entry : unknown ) {
s < < entry . first ;
s < < entry . second ;
}
s < < PSBT_SEPARATOR ;
}
template < typename Stream >
inline void Unserialize ( Stream & s ) {
// Read loop
while ( ! s . empty ( ) ) {
// Read
std : : vector < unsigned char > key ;
s > > key ;
// the key is empty if that was actually a separator byte
// This is a special case for key lengths 0 as those are not allowed (except for separator)
if ( key . empty ( ) ) return ;
// First byte of key is the type
unsigned char type = key [ 0 ] ;
// Do stuff based on type
switch ( type ) {
case PSBT_OUT_REDEEMSCRIPT :
{
if ( ! redeem_script . empty ( ) ) {
throw std : : ios_base : : failure ( " Duplicate Key, output redeemScript already provided " ) ;
2018-07-19 18:29:25 -07:00
} else if ( key . size ( ) ! = 1 ) {
throw std : : ios_base : : failure ( " Output redeemScript key is more than one byte type " ) ;
2018-06-27 16:53:48 -07:00
}
s > > redeem_script ;
break ;
}
case PSBT_OUT_WITNESSSCRIPT :
{
if ( ! witness_script . empty ( ) ) {
throw std : : ios_base : : failure ( " Duplicate Key, output witnessScript already provided " ) ;
2018-07-19 18:29:25 -07:00
} else if ( key . size ( ) ! = 1 ) {
throw std : : ios_base : : failure ( " Output witnessScript key is more than one byte type " ) ;
2018-06-27 16:53:48 -07:00
}
s > > witness_script ;
break ;
}
case PSBT_OUT_BIP32_DERIVATION :
{
DeserializeHDKeypaths ( s , key , hd_keypaths ) ;
break ;
}
// Unknown stuff
default : {
if ( unknown . count ( key ) > 0 ) {
throw std : : ios_base : : failure ( " Duplicate Key, key for unknown value already provided " ) ;
}
// Read in the value
std : : vector < unsigned char > val_bytes ;
s > > val_bytes ;
unknown . emplace ( std : : move ( key ) , std : : move ( val_bytes ) ) ;
break ;
}
}
}
}
template < typename Stream >
PSBTOutput ( deserialize_type , Stream & s ) {
Unserialize ( s ) ;
}
} ;
/** A version of CTransaction with the PSBT format*/
struct PartiallySignedTransaction
{
boost : : optional < CMutableTransaction > tx ;
std : : vector < PSBTInput > inputs ;
std : : vector < PSBTOutput > outputs ;
std : : map < std : : vector < unsigned char > , std : : vector < unsigned char > > unknown ;
bool IsNull ( ) const ;
2018-06-27 16:58:01 -07:00
void Merge ( const PartiallySignedTransaction & psbt ) ;
bool IsSane ( ) const ;
2018-06-27 16:53:48 -07:00
PartiallySignedTransaction ( ) { }
PartiallySignedTransaction ( const PartiallySignedTransaction & psbt_in ) : tx ( psbt_in . tx ) , inputs ( psbt_in . inputs ) , outputs ( psbt_in . outputs ) , unknown ( psbt_in . unknown ) { }
// Only checks if they refer to the same transaction
friend bool operator = = ( const PartiallySignedTransaction & a , const PartiallySignedTransaction & b )
{
return a . tx - > GetHash ( ) = = b . tx - > GetHash ( ) ;
}
friend bool operator ! = ( const PartiallySignedTransaction & a , const PartiallySignedTransaction & b )
{
return ! ( a = = b ) ;
}
template < typename Stream >
inline void Serialize ( Stream & s ) const {
// magic bytes
s < < PSBT_MAGIC_BYTES ;
// unsigned tx flag
SerializeToVector ( s , PSBT_GLOBAL_UNSIGNED_TX ) ;
// Write serialized tx to a stream
SerializeToVector ( s , * tx ) ;
// Write the unknown things
for ( auto & entry : unknown ) {
s < < entry . first ;
s < < entry . second ;
}
// Separator
s < < PSBT_SEPARATOR ;
// Write inputs
for ( const PSBTInput & input : inputs ) {
s < < input ;
}
// Write outputs
for ( const PSBTOutput & output : outputs ) {
s < < output ;
}
}
template < typename Stream >
inline void Unserialize ( Stream & s ) {
// Read the magic bytes
uint8_t magic [ 5 ] ;
s > > magic ;
if ( ! std : : equal ( magic , magic + 5 , PSBT_MAGIC_BYTES ) ) {
throw std : : ios_base : : failure ( " Invalid PSBT magic bytes " ) ;
}
// Read global data
while ( ! s . empty ( ) ) {
// Read
std : : vector < unsigned char > key ;
s > > key ;
// the key is empty if that was actually a separator byte
// This is a special case for key lengths 0 as those are not allowed (except for separator)
if ( key . empty ( ) ) break ;
// First byte of key is the type
unsigned char type = key [ 0 ] ;
// Do stuff based on type
switch ( type ) {
case PSBT_GLOBAL_UNSIGNED_TX :
{
if ( tx ) {
throw std : : ios_base : : failure ( " Duplicate Key, unsigned tx already provided " ) ;
2018-07-19 18:29:25 -07:00
} else if ( key . size ( ) ! = 1 ) {
throw std : : ios_base : : failure ( " Global unsigned tx key is more than one byte type " ) ;
2018-06-27 16:53:48 -07:00
}
CMutableTransaction mtx ;
UnserializeFromVector ( s , mtx ) ;
tx = std : : move ( mtx ) ;
// Make sure that all scriptSigs and scriptWitnesses are empty
for ( const CTxIn & txin : tx - > vin ) {
if ( ! txin . scriptSig . empty ( ) | | ! txin . scriptWitness . IsNull ( ) ) {
throw std : : ios_base : : failure ( " Unsigned tx does not have empty scriptSigs and scriptWitnesses. " ) ;
}
}
break ;
}
// Unknown stuff
default : {
if ( unknown . count ( key ) > 0 ) {
throw std : : ios_base : : failure ( " Duplicate Key, key for unknown value already provided " ) ;
}
// Read in the value
std : : vector < unsigned char > val_bytes ;
s > > val_bytes ;
unknown . emplace ( std : : move ( key ) , std : : move ( val_bytes ) ) ;
}
}
}
// Make sure that we got an unsigned tx
if ( ! tx ) {
throw std : : ios_base : : failure ( " No unsigned transcation was provided " ) ;
}
// Read input data
unsigned int i = 0 ;
while ( ! s . empty ( ) & & i < tx - > vin . size ( ) ) {
PSBTInput input ;
s > > input ;
inputs . push_back ( input ) ;
// Make sure the non-witness utxo matches the outpoint
if ( input . non_witness_utxo & & input . non_witness_utxo - > GetHash ( ) ! = tx - > vin [ i ] . prevout . hash ) {
throw std : : ios_base : : failure ( " Non-witness UTXO does not match outpoint hash " ) ;
}
+ + i ;
}
// Make sure that the number of inputs matches the number of inputs in the transaction
if ( inputs . size ( ) ! = tx - > vin . size ( ) ) {
throw std : : ios_base : : failure ( " Inputs provided does not match the number of inputs in transaction. " ) ;
}
// Read output data
i = 0 ;
while ( ! s . empty ( ) & & i < tx - > vout . size ( ) ) {
PSBTOutput output ;
s > > output ;
outputs . push_back ( output ) ;
+ + i ;
}
// Make sure that the number of outputs matches the number of outputs in the transaction
if ( outputs . size ( ) ! = tx - > vout . size ( ) ) {
throw std : : ios_base : : failure ( " Outputs provided does not match the number of outputs in transaction. " ) ;
}
2018-06-27 16:58:01 -07:00
// Sanity check
if ( ! IsSane ( ) ) {
throw std : : ios_base : : failure ( " PSBT is not sane. " ) ;
}
2018-06-27 16:53:48 -07:00
}
template < typename Stream >
PartiallySignedTransaction ( deserialize_type , Stream & s ) {
Unserialize ( s ) ;
}
} ;
2014-11-04 10:06:20 -08:00
/** Produce a script signature using a generic signature creator. */
2018-03-27 13:15:10 -07:00
bool ProduceSignature ( const SigningProvider & provider , const BaseSignatureCreator & creator , const CScript & scriptPubKey , SignatureData & sigdata ) ;
2014-11-04 10:06:20 -08:00
/** Produce a script signature for a transaction. */
2018-03-17 19:19:09 -07:00
bool SignSignature ( const SigningProvider & provider , const CScript & fromPubKey , CMutableTransaction & txTo , unsigned int nIn , const CAmount & amount , int nHashType ) ;
bool SignSignature ( const SigningProvider & provider , const CTransaction & txFrom , CMutableTransaction & txTo , unsigned int nIn , int nHashType ) ;
2014-08-27 17:22:33 +02:00
2018-08-08 14:18:55 -07:00
/** Signs a PSBTInput, verifying that all provided data matches what is being signed. */
2018-06-28 18:56:34 -07:00
bool SignPSBTInput ( const SigningProvider & provider , const CMutableTransaction & tx , PSBTInput & input , SignatureData & sigdata , int index , int sighash = 1 ) ;
2018-07-03 17:18:52 -07:00
/** Extract signature data from a transaction input, and insert it. */
SignatureData DataFromTransaction ( const CMutableTransaction & tx , unsigned int nIn , const CTxOut & txout ) ;
2018-03-05 16:37:24 -05:00
void UpdateInput ( CTxIn & input , const SignatureData & data ) ;
2014-08-27 17:22:33 +02:00
2017-11-30 16:48:31 -08:00
/* Check whether we know how to sign for an output like this, assuming we
* have all private keys . While this function does not need private keys , the passed
2018-03-17 19:19:09 -07:00
* provider is used to look up public keys and redeemscripts by hash .
2017-11-30 16:48:31 -08:00
* Solvability is unrelated to whether we consider this output to be ours . */
2018-03-17 19:19:09 -07:00
bool IsSolvable ( const SigningProvider & provider , const CScript & script ) ;
2017-11-30 16:48:31 -08:00
2014-11-03 16:16:40 +01:00
# endif // BITCOIN_SCRIPT_SIGN_H