@ -13,31 +13,39 @@ function isCoinbase(txHash) {
} )
} )
}
}
function extractSignature ( txIn ) {
function extractInput ( txIn ) {
var redeemScript
var redeemScript
var scriptSig = txIn . script
var scriptSig = txIn . script
var scriptType = scripts . classifyInput ( scriptSig , true )
var prevOutScript
var prevOutType = scripts . classifyInput ( scriptSig , true )
var scriptType
// Re-classify if P2SH
// Re-classify if P2SH
if ( scrip tType === 'scripthash' ) {
if ( prevOu tType === 'scripthash' ) {
redeemScript = Script . fromBuffer ( scriptSig . chunks . slice ( - 1 ) [ 0 ] )
redeemScript = Script . fromBuffer ( scriptSig . chunks . slice ( - 1 ) [ 0 ] )
scriptSig = Script . fromChunks ( scriptSig . chunks . slice ( 0 , - 1 ) )
prevOutScript = scripts . scriptHashOutput ( redeemScript . getHash ( ) )
scriptSig = Script . fromChunks ( scriptSig . chunks . slice ( 0 , - 1 ) )
scriptType = scripts . classifyInput ( scriptSig , true )
scriptType = scripts . classifyInput ( scriptSig , true )
assert . equal ( scripts . classifyOutput ( redeemScript ) , scriptType , 'Non-matching scriptSig and scriptPubKey in input' )
assert . equal ( scripts . classifyOutput ( redeemScript ) , scriptType , 'Non-matching scriptSig and scriptPubKey in input' )
} else {
scriptType = prevOutType
}
}
// Extract hashType, pubKeys and signatures
// Extract hashType, pubKeys and signatures
var hashType , parsed , pubKeys , signatures
var hashType , initialized , parsed , pubKeys , signatures
switch ( scriptType ) {
switch ( scriptType ) {
case 'pubkeyhash' :
case 'pubkeyhash' :
parsed = ECSignature . parseScriptSignature ( scriptSig . chunks [ 0 ] )
parsed = ECSignature . parseScriptSignature ( scriptSig . chunks [ 0 ] )
hashType = parsed . hashType
hashType = parsed . hashType
pubKeys = [ ECPubKey . fromBuffer ( scriptSig . chunks [ 1 ] ) ]
pubKeys = [ ECPubKey . fromBuffer ( scriptSig . chunks [ 1 ] ) ]
signatures = [ parsed . signature ]
signatures = [ parsed . signature ]
initialized = true
prevOutScript = pubKeys [ 0 ] . getAddress ( ) . toOutputScript ( )
break
break
case 'multisig' :
case 'multisig' :
@ -46,8 +54,8 @@ function extractSignature(txIn) {
} ) . map ( ECSignature . parseScriptSignature )
} ) . map ( ECSignature . parseScriptSignature )
hashType = parsed [ 0 ] . hashType
hashType = parsed [ 0 ] . hashType
pubKeys = [ ]
signatures = parsed . map ( function ( p ) { return p . signature } )
signatures = parsed . map ( function ( p ) { return p . signature } )
initialized = true
if ( redeemScript ) {
if ( redeemScript ) {
pubKeys = redeemScript . chunks . slice ( 1 , - 2 ) . map ( ECPubKey . fromBuffer )
pubKeys = redeemScript . chunks . slice ( 1 , - 2 ) . map ( ECPubKey . fromBuffer )
@ -57,11 +65,11 @@ function extractSignature(txIn) {
case 'pubkey' :
case 'pubkey' :
parsed = ECSignature . parseScriptSignature ( scriptSig . chunks [ 0 ] )
parsed = ECSignature . parseScriptSignature ( scriptSig . chunks [ 0 ] )
hashType = parsed . hashType
hashType = parsed . hashType
pubKeys = [ ]
signatures = [ parsed . signature ]
signatures = [ parsed . signature ]
initialized = true
if ( redeemScript ) {
if ( redeemScript ) {
pubKeys = [ ECPubKey . fromBuffer ( redeemScript . chunks [ 0 ] ) ]
pubKeys = [ ECPubKey . fromBuffer ( redeemScript . chunks [ 0 ] ) ]
}
}
@ -69,11 +77,18 @@ function extractSignature(txIn) {
break
break
default :
default :
assert ( false , scriptType + ' inputs not supported' )
if ( redeemScript ) {
initialized = true
}
break
}
}
return {
return {
hashType : hashType ,
hashType : hashType ,
initialized : initialized ,
prevOutScript : prevOutScript ,
prevOutType : prevOutType ,
pubKeys : pubKeys ,
pubKeys : pubKeys ,
redeemScript : redeemScript ,
redeemScript : redeemScript ,
scriptType : scriptType ,
scriptType : scriptType ,
@ -86,7 +101,7 @@ function TransactionBuilder() {
this . prevOutScripts = { }
this . prevOutScripts = { }
this . prevOutTypes = { }
this . prevOutTypes = { }
this . signature s = [ ]
this . input s = [ ]
this . tx = new Transaction ( )
this . tx = new Transaction ( )
}
}
@ -108,14 +123,16 @@ TransactionBuilder.fromTransaction = function(transaction) {
} )
} )
// Extract/add signatures
// Extract/add signatures
txb . signatures = transaction . ins . map ( function ( txIn ) {
txb . inputs = transaction . ins . map ( function ( txIn ) {
// TODO: remove me after testcase added
// Coinbase inputs not supported
assert ( ! isCoinbase ( txIn . hash ) , 'coinbase inputs not supported' )
assert ( ! Array . prototype . every . call ( txIn . hash , function ( x ) {
return x === 0
} ) , 'coinbase inputs not supported' )
// Ignore empty scripts
// Ignore empty scripts
if ( txIn . script . buffer . length === 0 ) return
if ( txIn . script . buffer . length === 0 ) return
return extractSignature ( txIn )
return extractInput ( txIn )
} )
} )
return txb
return txb
@ -139,31 +156,51 @@ TransactionBuilder.prototype.addInput = function(prevTx, index, sequence, prevOu
}
}
var prevOutType
var input = { }
if ( prevOutScript !== undefined ) {
if ( prevOutScript ) {
prevOutType = scripts . classifyOutput ( prevOutScript )
var prevOutType = scripts . classifyOutput ( prevOutScript )
// if we can, extract pubKey information
switch ( prevOutType ) {
case 'multisig' :
input . pubKeys = prevOutScript . chunks . slice ( 1 , - 2 ) . map ( ECPubKey . fromBuffer )
break
assert . notEqual ( prevOutType , 'nonstandard' , 'PrevOutScript not supported (nonstandard)' )
case 'pubkey' :
input . pubKeys = prevOutScript . chunks . slice ( 0 , 1 ) . map ( ECPubKey . fromBuffer )
break
}
}
assert ( this . signatures . every ( function ( input ) {
if ( prevOutType !== 'scripthash' ) {
return input . hashType & Transaction . SIGHASH_ANYONECANPAY
input . scriptType = prevOutType
}
input . prevOutScript = prevOutScript
input . prevOutType = prevOutType
}
assert ( this . inputs . every ( function ( input2 ) {
if ( input2 . hashType === undefined ) return true
return input2 . hashType & Transaction . SIGHASH_ANYONECANPAY
} ) , 'No, this would invalidate signatures' )
} ) , 'No, this would invalidate signatures' )
var prevOut = prevOutHash . toString ( 'hex' ) + ':' + index
var prevOut = prevOutHash . toString ( 'hex' ) + ':' + index
assert ( ! ( prevOut in this . prevTxMap ) , 'Transaction is already an input' )
assert ( ! ( prevOut in this . prevTxMap ) , 'Transaction is already an input' )
var vout = this . tx . addInput ( prevOutHash , index , sequence )
var vout = this . tx . addInput ( prevOutHash , index , sequence )
this . prevTxMap [ prevOut ] = true
this . prevTxMap [ prevOut ] = true
this . prevOutScripts [ vout ] = prevOutScript
this . inputs [ vout ] = input
this . prevOutTypes [ vout ] = prevOutType
return vout
return vout
}
}
TransactionBuilder . prototype . addOutput = function ( scriptPubKey , value ) {
TransactionBuilder . prototype . addOutput = function ( scriptPubKey , value ) {
assert ( this . signatures . every ( function ( signature ) {
assert ( this . inputs . every ( function ( input ) {
return ( signature . hashType & 0x1f ) === Transaction . SIGHASH_SINGLE
if ( input . hashType === undefined ) return true
return ( input . hashType & 0x1f ) === Transaction . SIGHASH_SINGLE
} ) , 'No, this would invalidate signatures' )
} ) , 'No, this would invalidate signatures' )
return this . tx . addOutput ( scriptPubKey , value )
return this . tx . addOutput ( scriptPubKey , value )
@ -175,41 +212,50 @@ TransactionBuilder.prototype.__build = function(allowIncomplete) {
if ( ! allowIncomplete ) {
if ( ! allowIncomplete ) {
assert ( this . tx . ins . length > 0 , 'Transaction has no inputs' )
assert ( this . tx . ins . length > 0 , 'Transaction has no inputs' )
assert ( this . tx . outs . length > 0 , 'Transaction has no outputs' )
assert ( this . tx . outs . length > 0 , 'Transaction has no outputs' )
assert ( this . signatures . length > 0 , 'Transaction has no signatures' )
assert . equal ( this . signatures . length , this . tx . ins . length , 'Transaction is missing signatures' )
}
}
var tx = this . tx . clone ( )
var tx = this . tx . clone ( )
// Create script signatures from signature meta-data
// Create script signatures from signature meta-data
this . signatures . forEach ( function ( input , index ) {
this . inputs . forEach ( function ( input , index ) {
var scriptSig
if ( ! allowIncomplete ) {
var scriptType = input . scriptType
assert ( input . initialized , 'Transaction is not complete' )
}
var signatures = input . signatures . map ( function ( signature ) {
var scriptSig
return signature . toScriptSignature ( input . hashType )
} )
switch ( scriptType ) {
switch ( input . scriptType ) {
case 'pubkeyhash' :
case 'pubkeyhash' :
var pubKey = input . pubKeys [ 0 ]
assert ( input . signatures , 'Transaction is missing signatures' )
scriptSig = scripts . pubKeyHashInput ( signatures [ 0 ] , pubKey )
assert . equal ( input . signatures . length , 1 , 'Transaction is missing signatures' )
var pkhSignature = input . signatures [ 0 ] . toScriptSignature ( input . hashType )
scriptSig = scripts . pubKeyHashInput ( pkhSignature , input . pubKeys [ 0 ] )
break
break
case 'multisig' :
case 'multisig' :
assert ( input . signatures , 'Transaction is missing signatures' )
var signatures = input . signatures . map ( function ( signature ) {
return signature . toScriptSignature ( input . hashType )
} ) . filter ( function ( signature ) { return ! ! signature } )
var redeemScript = allowIncomplete ? undefined : input . redeemScript
var redeemScript = allowIncomplete ? undefined : input . redeemScript
scriptSig = scripts . multisigInput ( signatures , redeemScript )
scriptSig = scripts . multisigInput ( signatures , redeemScript )
break
break
case 'pubkey' :
case 'pubkey' :
scriptSig = scripts . pubKeyInput ( signatures [ 0 ] )
assert ( input . signatures , 'Transaction is missing signatures' )
assert . equal ( input . signatures . length , 1 , 'Transaction is missing signatures' )
var pkSignature = input . signatures [ 0 ] . toScriptSignature ( input . hashType )
scriptSig = scripts . pubKeyInput ( pkSignature )
break
break
default :
default :
assert ( false , scriptType + ' not supported' )
if ( allowIncomplete ) return
assert ( false , input . scriptType + ' not supported' )
}
}
if ( input . redeemScript ) {
if ( input . redeemScript ) {
@ -223,73 +269,116 @@ TransactionBuilder.prototype.__build = function(allowIncomplete) {
}
}
TransactionBuilder . prototype . sign = function ( index , privKey , redeemScript , hashType ) {
TransactionBuilder . prototype . sign = function ( index , privKey , redeemScript , hashType ) {
assert ( this . tx . ins . length >= index , 'No input at index: ' + index )
assert ( index in this . inputs , 'No input at index: ' + index )
hashType = hashType || Transaction . SIGHASH_ALL
hashType = hashType || Transaction . SIGHASH_ALL
var prevOutScript = this . prevOutScripts [ index ]
var input = this . inputs [ index ]
var prevOutType = this . prevOutTypes [ index ]
var scriptType , hash
if ( input . hashType !== undefined ) {
if ( redeemScript ) {
assert . equal ( input . hashType , hashType , 'Inconsistent hashType' )
prevOutScript = prevOutScript || scripts . scriptHashOutput ( redeemScript . getHash ( ) )
}
prevOutType = prevOutType || 'scripthash'
assert . equal ( prevOutType , 'scripthash' , 'PrevOutScript must be P2SH' )
// are we already initialized?
if ( input . initialized ) {
if ( input . prevOutType === 'scripthash' ) {
assert ( input . redeemScript , 'PrevOutScript is P2SH, missing redeemScript' )
scriptType = scripts . classifyOutput ( redeemScript )
} else {
assert ( ! input . redeemScript , 'PrevOutScript must be P2SH' )
}
assert . notEqual ( scriptType , 'scripthash' , 'RedeemScript can\'t be P2SH' )
// redeemScript only needed to initialize, but if provided again, enforce consistency
assert . notEqual ( scriptType , 'nulldata' , 'RedeemScript not supported (nulldata)' )
if ( redeemScript ) {
assert . notEqual ( scriptType , 'nonstandard' , 'RedeemScript not supported (nonstandard)' )
assert . deepEqual ( input . redeemScript , redeemScript , 'Inconsistent redeemScript' )
}
hash = this . tx . hashForSignature ( index , redeemScript , hashType )
// if signatures already exist, enforce multisig scriptType
if ( input . signatures . length > 0 ) {
assert . equal ( input . scriptType , 'multisig' , input . scriptType + ' doesn\'t support multiple signatures' )
}
// initialize it
} else {
} else {
prevOutScript = prevOutScript || privKey . pub . getAddress ( ) . toOutputScript ( )
if ( redeemScript ) {
prevOutType = prevOutType || 'pubkeyhash'
if ( input . prevOutScript ) {
assert . equal ( input . prevOutType , 'scripthash' , 'PrevOutScript must be P2SH' )
assert . notEqual ( prevOutType , 'scripthash' , 'PrevOutScript is P2SH, missing redeemScript' )
scriptType = prevOutType
var scriptHash = input . prevOutScript . chunks [ 1 ]
assert . deepEqual ( scriptHash , redeemScript . getHash ( ) , 'RedeemScript does not match ' + scriptHash . toString ( 'hex' ) )
hash = this . tx . hashForSignature ( index , prevOutScript , hashType )
} else {
input . prevOutScript = scripts . scriptHashOutput ( redeemScript . getHash ( ) )
input . prevOutType = 'scripthash'
}
}
var input = this . signatures [ index ]
var scriptType = scripts . classifyOutput ( redeemScript )
if ( ! input ) {
var pubKeys = [ ]
if ( redeemScript && scriptType === 'multisig' ) {
switch ( scriptType ) {
pubKeys = redeemScript . chunks . slice ( 1 , - 2 ) . map ( ECPubKey . fromBuffer )
case 'multisig' :
input . pubKeys = redeemScript . chunks . slice ( 1 , - 2 ) . map ( ECPubKey . fromBuffer )
break
case 'pubkeyhash' :
var pkh1 = redeemScript . chunks [ 2 ]
var pkh2 = privKey . pub . getAddress ( ) . hash
assert . deepEqual ( pkh1 , pkh2 , 'privateKey cannot sign for this input' )
input . pubKeys = [ privKey . pub ]
break
case 'pubkey' :
input . pubKeys = redeemScript . chunks . slice ( 0 , 1 ) . map ( ECPubKey . fromBuffer )
break
default :
assert ( false , 'RedeemScript not supported (' + scriptType + ')' )
}
input . redeemScript = redeemScript
input . scriptType = scriptType
} else {
} else {
pubKeys . push ( privKey . pub )
assert . notEqual ( input . prevOutType , 'scripthash' , 'PrevOutScript is P2SH, missing redeemScript' )
if ( ! input . scriptType ) {
input . prevOutScript = privKey . pub . getAddress ( ) . toOutputScript ( )
input . prevOutType = 'pubkeyhash'
input . pubKeys = [ privKey . pub ]
input . scriptType = input . prevOutType
}
}
}
input = {
input . hashType = hashType
hashType : hashType ,
input . initialized = true
pubKeys : pubKeys ,
input . signatures = input . signatures || [ ]
redeemScript : redeemScript ,
scriptType : scriptType ,
signatures : [ ]
}
}
this . signatures [ index ] = input
switch ( input . scriptType ) {
this . prevOutScripts [ index ] = prevOutScript
case 'pubkeyhash' :
this . prevOutTypes [ index ] = prevOutType
case 'multisig' :
case 'pubkey' :
break
default :
assert ( false , input . scriptType + ' not supported' )
}
var signatureHash
if ( input . redeemScript ) {
signatureHash = this . tx . hashForSignature ( index , input . redeemScript , hashType )
} else {
} else {
assert . equal ( scriptType , 'multisig' , scriptType + ' doesn\'t support multiple signatures' )
signatureHash = this . tx . hashForSignature ( index , input . prevOutScript , hashType )
assert . equal ( input . hashType , hashType , 'Inconsistent hashType' )
assert . deepEqual ( input . redeemScript , redeemScript , 'Inconsistent redeemScript' )
}
}
var signature = privKey . sign ( signatureHash )
// enforce signing in order of public keys
// enforce signing in order of public keys
assert ( input . pubKeys . some ( function ( pubKey , i ) {
assert ( input . pubKeys . some ( function ( pubKey , i ) {
if ( ! privKey . pub . Q . equals ( pubKey . Q ) ) return false // FIXME: could be better?
if ( ! privKey . pub . Q . equals ( pubKey . Q ) ) return false
assert ( ! input . signatures [ i ] , 'Signature already exists' )
assert ( ! input . signatures [ i ] , 'Signature already exists' )
input . signatures [ i ] = privKey . sign ( hash )
input . signatures [ i ] = signature
return true
return true
} ) , 'privateKey cannot sign for this input' )
} ) , 'privateKey cannot sign for this input' )