2015-12-13 14:51:43 +01:00
// Copyright (c) 2009-2015 The Bitcoin Core developers
2014-12-13 05:09:33 +01:00
// Distributed under the MIT software license, see the accompanying
2014-08-01 08:39:06 +02:00
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
2014-06-24 05:10:24 +02:00
# include "core_io.h"
2014-09-14 12:43:56 +02:00
# include "base58.h"
2014-11-18 22:03:02 +01:00
# include "primitives/transaction.h"
2014-08-23 03:35:51 +02:00
# include "script/script.h"
# include "script/standard.h"
2014-06-24 05:10:24 +02:00
# include "serialize.h"
2014-10-22 21:08:30 +02:00
# include "streams.h"
2019-06-11 13:42:17 +02:00
# include "univalue.h"
2014-06-24 05:10:24 +02:00
# include "util.h"
Split up util.cpp/h
Split up util.cpp/h into:
- string utilities (hex, base32, base64): no internal dependencies, no dependency on boost (apart from foreach)
- money utilities (parsesmoney, formatmoney)
- time utilities (gettime*, sleep, format date):
- and the rest (logging, argument parsing, config file parsing)
The latter is basically the environment and OS handling,
and is stripped of all utility functions, so we may want to
rename it to something else than util.cpp/h for clarity (Matt suggested
osinterface).
Breaks dependency of sha256.cpp on all the things pulled in by util.
2014-08-21 16:11:09 +02:00
# include "utilmoneystr.h"
2014-09-25 05:32:36 +02:00
# include "utilstrencodings.h"
2014-06-24 05:10:24 +02:00
2019-06-11 13:42:17 +02:00
# include "spentindex.h"
# include "evo/cbtx.h"
# include "evo/providertx.h"
# include "evo/specialtx.h"
# include "llmq/quorums_commitment.h"
2015-07-31 01:56:00 +02:00
# include <boost/assign/list_of.hpp>
2014-08-23 05:09:47 +02:00
# include <boost/foreach.hpp>
2017-01-30 13:13:07 +01:00
std : : string FormatScript ( const CScript & script )
2014-09-20 03:13:04 +02:00
{
2017-01-30 13:13:07 +01:00
std : : string ret ;
2014-09-20 03:13:04 +02:00
CScript : : const_iterator it = script . begin ( ) ;
opcodetype op ;
while ( it ! = script . end ( ) ) {
CScript : : const_iterator it2 = it ;
2017-01-30 13:13:07 +01:00
std : : vector < unsigned char > vch ;
2014-09-20 03:13:04 +02:00
if ( script . GetOp2 ( it , op , & vch ) ) {
if ( op = = OP_0 ) {
ret + = " 0 " ;
continue ;
} else if ( ( op > = OP_1 & & op < = OP_16 ) | | op = = OP_1NEGATE ) {
ret + = strprintf ( " %i " , op - OP_1NEGATE - 1 ) ;
continue ;
2016-04-05 14:54:18 +02:00
} else if ( op > = OP_NOP & & op < = OP_NOP10 ) {
2017-01-30 13:13:07 +01:00
std : : string str ( GetOpName ( op ) ) ;
if ( str . substr ( 0 , 3 ) = = std : : string ( " OP_ " ) ) {
ret + = str . substr ( 3 , std : : string : : npos ) + " " ;
2014-09-20 03:13:04 +02:00
continue ;
}
}
if ( vch . size ( ) > 0 ) {
ret + = strprintf ( " 0x%x 0x%x " , HexStr ( it2 , it - vch . size ( ) ) , HexStr ( it - vch . size ( ) , it ) ) ;
} else {
2016-04-05 14:54:18 +02:00
ret + = strprintf ( " 0x%x " , HexStr ( it2 , it ) ) ;
2014-09-20 03:13:04 +02:00
}
continue ;
}
ret + = strprintf ( " 0x%x " , HexStr ( it2 , script . end ( ) ) ) ;
break ;
}
return ret . substr ( 0 , ret . size ( ) - 1 ) ;
}
2017-01-30 13:13:07 +01:00
const std : : map < unsigned char , std : : string > mapSigHashTypes =
2015-07-31 01:56:00 +02:00
boost : : assign : : map_list_of
2017-01-30 13:13:07 +01:00
( static_cast < unsigned char > ( SIGHASH_ALL ) , std : : string ( " ALL " ) )
( static_cast < unsigned char > ( SIGHASH_ALL | SIGHASH_ANYONECANPAY ) , std : : string ( " ALL|ANYONECANPAY " ) )
( static_cast < unsigned char > ( SIGHASH_NONE ) , std : : string ( " NONE " ) )
( static_cast < unsigned char > ( SIGHASH_NONE | SIGHASH_ANYONECANPAY ) , std : : string ( " NONE|ANYONECANPAY " ) )
( static_cast < unsigned char > ( SIGHASH_SINGLE ) , std : : string ( " SINGLE " ) )
( static_cast < unsigned char > ( SIGHASH_SINGLE | SIGHASH_ANYONECANPAY ) , std : : string ( " SINGLE|ANYONECANPAY " ) )
2015-07-31 01:56:00 +02:00
;
/**
* Create the assembly string representation of a CScript object .
* @ param [ in ] script CScript object to convert into the asm string representation .
* @ param [ in ] fAttemptSighashDecode Whether to attempt to decode sighash types on data within the script that matches the format
* of a signature . Only pass true for scripts you believe could contain signatures . For example ,
* pass false , or omit the this argument ( defaults to false ) , for scriptPubKeys .
*/
2017-01-30 13:13:07 +01:00
std : : string ScriptToAsmStr ( const CScript & script , const bool fAttemptSighashDecode )
2015-07-31 01:56:00 +02:00
{
2017-01-30 13:13:07 +01:00
std : : string str ;
2015-07-31 01:56:00 +02:00
opcodetype opcode ;
2017-01-30 13:13:07 +01:00
std : : vector < unsigned char > vch ;
2015-07-31 01:56:00 +02:00
CScript : : const_iterator pc = script . begin ( ) ;
while ( pc < script . end ( ) ) {
if ( ! str . empty ( ) ) {
str + = " " ;
}
if ( ! script . GetOp ( pc , opcode , vch ) ) {
str + = " [error] " ;
return str ;
}
if ( 0 < = opcode & & opcode < = OP_PUSHDATA4 ) {
2017-01-30 13:13:07 +01:00
if ( vch . size ( ) < = static_cast < std : : vector < unsigned char > : : size_type > ( 4 ) ) {
2015-07-31 01:56:00 +02:00
str + = strprintf ( " %d " , CScriptNum ( vch , false ) . getint ( ) ) ;
} else {
// the IsUnspendable check makes sure not to try to decode OP_RETURN data that may match the format of a signature
if ( fAttemptSighashDecode & & ! script . IsUnspendable ( ) ) {
2017-01-30 13:13:07 +01:00
std : : string strSigHashDecode ;
2015-07-31 01:56:00 +02:00
// goal: only attempt to decode a defined sighash type from data that looks like a signature within a scriptSig.
// this won't decode correctly formatted public keys in Pubkey or Multisig scripts due to
// the restrictions on the pubkey formats (see IsCompressedOrUncompressedPubKey) being incongruous with the
// checks in CheckSignatureEncoding.
if ( CheckSignatureEncoding ( vch , SCRIPT_VERIFY_STRICTENC , NULL ) ) {
const unsigned char chSigHashType = vch . back ( ) ;
if ( mapSigHashTypes . count ( chSigHashType ) ) {
strSigHashDecode = " [ " + mapSigHashTypes . find ( chSigHashType ) - > second + " ] " ;
vch . pop_back ( ) ; // remove the sighash type byte. it will be replaced by the decode.
}
}
str + = HexStr ( vch ) + strSigHashDecode ;
} else {
str + = HexStr ( vch ) ;
}
}
} else {
str + = GetOpName ( opcode ) ;
}
}
return str ;
}
2017-01-30 13:13:07 +01:00
std : : string EncodeHexTx ( const CTransaction & tx )
2014-06-24 05:10:24 +02:00
{
CDataStream ssTx ( SER_NETWORK , PROTOCOL_VERSION ) ;
ssTx < < tx ;
return HexStr ( ssTx . begin ( ) , ssTx . end ( ) ) ;
}
2014-07-29 17:12:44 +02:00
void ScriptPubKeyToUniv ( const CScript & scriptPubKey ,
UniValue & out , bool fIncludeHex )
{
txnouttype type ;
2017-01-30 13:13:07 +01:00
std : : vector < CTxDestination > addresses ;
2014-07-29 17:12:44 +02:00
int nRequired ;
2015-07-31 01:56:00 +02:00
out . pushKV ( " asm " , ScriptToAsmStr ( scriptPubKey ) ) ;
2014-07-29 17:12:44 +02:00
if ( fIncludeHex )
out . pushKV ( " hex " , HexStr ( scriptPubKey . begin ( ) , scriptPubKey . end ( ) ) ) ;
if ( ! ExtractDestinations ( scriptPubKey , type , addresses , nRequired ) ) {
out . pushKV ( " type " , GetTxnOutputType ( type ) ) ;
return ;
}
out . pushKV ( " reqSigs " , nRequired ) ;
out . pushKV ( " type " , GetTxnOutputType ( type ) ) ;
UniValue a ( UniValue : : VARR ) ;
BOOST_FOREACH ( const CTxDestination & addr , addresses )
a . push_back ( CBitcoinAddress ( addr ) . ToString ( ) ) ;
out . pushKV ( " addresses " , a ) ;
}
2019-06-11 13:42:17 +02:00
void TxToUniv ( const CTransaction & tx , const uint256 & hashBlock , UniValue & entry , const CSpentIndexTxInfo * ptxSpentInfo )
2014-07-29 17:12:44 +02:00
{
2019-06-11 13:42:17 +02:00
uint256 txid = tx . GetHash ( ) ;
entry . pushKV ( " txid " , txid . GetHex ( ) ) ;
2014-07-29 17:12:44 +02:00
entry . pushKV ( " version " , tx . nVersion ) ;
2019-06-11 13:42:17 +02:00
entry . pushKV ( " size " , ( int ) : : GetSerializeSize ( tx , SER_NETWORK , PROTOCOL_VERSION ) ) ;
2014-07-29 17:12:44 +02:00
entry . pushKV ( " locktime " , ( int64_t ) tx . nLockTime ) ;
UniValue vin ( UniValue : : VARR ) ;
BOOST_FOREACH ( const CTxIn & txin , tx . vin ) {
UniValue in ( UniValue : : VOBJ ) ;
if ( tx . IsCoinBase ( ) )
in . pushKV ( " coinbase " , HexStr ( txin . scriptSig . begin ( ) , txin . scriptSig . end ( ) ) ) ;
else {
in . pushKV ( " txid " , txin . prevout . hash . GetHex ( ) ) ;
in . pushKV ( " vout " , ( int64_t ) txin . prevout . n ) ;
UniValue o ( UniValue : : VOBJ ) ;
2015-07-31 01:56:00 +02:00
o . pushKV ( " asm " , ScriptToAsmStr ( txin . scriptSig , true ) ) ;
2014-07-29 17:12:44 +02:00
o . pushKV ( " hex " , HexStr ( txin . scriptSig . begin ( ) , txin . scriptSig . end ( ) ) ) ;
in . pushKV ( " scriptSig " , o ) ;
2019-06-11 13:42:17 +02:00
// Add address and value info if spentindex enabled
if ( ptxSpentInfo ! = nullptr ) {
CSpentIndexKey spentKey ( txin . prevout . hash , txin . prevout . n ) ;
auto it = ptxSpentInfo - > mSpentInfo . find ( spentKey ) ;
if ( it ! = ptxSpentInfo - > mSpentInfo . end ( ) ) {
auto spentInfo = it - > second ;
UniValue outValue ( UniValue : : VNUM , FormatMoney ( spentInfo . satoshis ) ) ;
in . push_back ( Pair ( " value " , outValue ) ) ;
in . push_back ( Pair ( " valueSat " , spentInfo . satoshis ) ) ;
if ( spentInfo . addressType = = 1 ) {
in . push_back ( Pair ( " address " , CBitcoinAddress ( CKeyID ( spentInfo . addressHash ) ) . ToString ( ) ) ) ;
} else if ( spentInfo . addressType = = 2 ) {
in . push_back ( Pair ( " address " , CBitcoinAddress ( CScriptID ( spentInfo . addressHash ) ) . ToString ( ) ) ) ;
}
}
}
2014-07-29 17:12:44 +02:00
}
in . pushKV ( " sequence " , ( int64_t ) txin . nSequence ) ;
vin . push_back ( in ) ;
}
entry . pushKV ( " vin " , vin ) ;
UniValue vout ( UniValue : : VARR ) ;
for ( unsigned int i = 0 ; i < tx . vout . size ( ) ; i + + ) {
const CTxOut & txout = tx . vout [ i ] ;
UniValue out ( UniValue : : VOBJ ) ;
UniValue outValue ( UniValue : : VNUM , FormatMoney ( txout . nValue ) ) ;
out . pushKV ( " value " , outValue ) ;
2019-06-11 13:42:17 +02:00
out . pushKV ( " valueSat " , txout . nValue ) ;
2014-07-29 17:12:44 +02:00
out . pushKV ( " n " , ( int64_t ) i ) ;
UniValue o ( UniValue : : VOBJ ) ;
ScriptPubKeyToUniv ( txout . scriptPubKey , o , true ) ;
out . pushKV ( " scriptPubKey " , o ) ;
2019-06-11 13:42:17 +02:00
// Add spent information if spentindex is enabled
if ( ptxSpentInfo ! = nullptr ) {
CSpentIndexKey spentKey ( txid , i ) ;
auto it = ptxSpentInfo - > mSpentInfo . find ( spentKey ) ;
if ( it ! = ptxSpentInfo - > mSpentInfo . end ( ) ) {
auto spentInfo = it - > second ;
out . push_back ( Pair ( " spentTxId " , spentInfo . txid . GetHex ( ) ) ) ;
out . push_back ( Pair ( " spentIndex " , ( int ) spentInfo . inputIndex ) ) ;
out . push_back ( Pair ( " spentHeight " , spentInfo . blockHeight ) ) ;
}
}
2014-07-29 17:12:44 +02:00
vout . push_back ( out ) ;
}
entry . pushKV ( " vout " , vout ) ;
2019-06-11 13:42:17 +02:00
if ( ! tx . vExtraPayload . empty ( ) ) {
entry . push_back ( Pair ( " extraPayloadSize " , ( int ) tx . vExtraPayload . size ( ) ) ) ;
entry . push_back ( Pair ( " extraPayload " , HexStr ( tx . vExtraPayload ) ) ) ;
}
if ( tx . nType = = TRANSACTION_PROVIDER_REGISTER ) {
CProRegTx proTx ;
if ( GetTxPayload ( tx , proTx ) ) {
UniValue obj ;
proTx . ToJson ( obj ) ;
entry . push_back ( Pair ( " proRegTx " , obj ) ) ;
}
} else if ( tx . nType = = TRANSACTION_PROVIDER_UPDATE_SERVICE ) {
CProUpServTx proTx ;
if ( GetTxPayload ( tx , proTx ) ) {
UniValue obj ;
proTx . ToJson ( obj ) ;
entry . push_back ( Pair ( " proUpServTx " , obj ) ) ;
}
} else if ( tx . nType = = TRANSACTION_PROVIDER_UPDATE_REGISTRAR ) {
CProUpRegTx proTx ;
if ( GetTxPayload ( tx , proTx ) ) {
UniValue obj ;
proTx . ToJson ( obj ) ;
entry . push_back ( Pair ( " proUpRegTx " , obj ) ) ;
}
} else if ( tx . nType = = TRANSACTION_PROVIDER_UPDATE_REVOKE ) {
CProUpRevTx proTx ;
if ( GetTxPayload ( tx , proTx ) ) {
UniValue obj ;
proTx . ToJson ( obj ) ;
entry . push_back ( Pair ( " proUpRevTx " , obj ) ) ;
}
} else if ( tx . nType = = TRANSACTION_COINBASE ) {
CCbTx cbTx ;
if ( GetTxPayload ( tx , cbTx ) ) {
UniValue obj ;
cbTx . ToJson ( obj ) ;
entry . push_back ( Pair ( " cbTx " , obj ) ) ;
}
} else if ( tx . nType = = TRANSACTION_QUORUM_COMMITMENT ) {
llmq : : CFinalCommitmentTxPayload qcTx ;
if ( GetTxPayload ( tx , qcTx ) ) {
UniValue obj ;
qcTx . ToJson ( obj ) ;
entry . push_back ( Pair ( " qcTx " , obj ) ) ;
}
}
2014-12-15 09:11:16 +01:00
if ( ! hashBlock . IsNull ( ) )
2014-07-29 17:12:44 +02:00
entry . pushKV ( " blockhash " , hashBlock . GetHex ( ) ) ;
2014-11-04 19:01:41 +01:00
entry . pushKV ( " hex " , EncodeHexTx ( tx ) ) ; // the hex-encoded transaction. used the name "hex" to be consistent with the verbose output of "getrawtransaction".
2014-07-29 17:12:44 +02:00
}