Skip to content

Commit

Permalink
PayToScriptHash support (WIP)
Browse files Browse the repository at this point in the history
  • Loading branch information
matiu committed Apr 1, 2014
1 parent 4fe8dff commit 4edab24
Show file tree
Hide file tree
Showing 4 changed files with 189 additions and 10 deletions.
84 changes: 74 additions & 10 deletions TransactionBuilder.js
Original file line number Diff line number Diff line change
Expand Up @@ -152,17 +152,35 @@ TransactionBuilder._scriptForPubkeys = function(out) {
};

TransactionBuilder._scriptForOut = function(out) {
var ret;
if (out.address)
ret = this._scriptForAddress(out.address)
var ret;
if (out.address)
ret = this._scriptForAddress(out.address);
else if (out.pubkeys || out.nreq || out.nreq > 1)
ret = this._scriptForPubkeys(out);
else
else
throw new Error('unknown out type');

return ret;
};


TransactionBuilder.infoForP2sh = function(opts, networkName) {
var script = this._scriptForOut(opts);
var hash = util.sha256ripe160(script.getBuffer());

var version = networkName === 'testnet' ?
networks.testnet.addressScript : networks.livenet.addressScript;

var addr = new Address(version, hash);
var addrStr = addr.as('base58');
return {
script: script,
scriptBufHex: script.getBuffer().toString('hex'),
hash: hash,
address: addrStr,
};
};

TransactionBuilder.prototype.setUnspent = function(utxos) {
this.utxos = utxos;
return this;
Expand All @@ -173,8 +191,7 @@ TransactionBuilder.prototype._setInputMap = function() {

var l = this.selectedUtxos.length;
for (var i = 0; i < l; i++) {
var utxo = this.selectedUtxos[i];

var utxo = this.selectedUtxos[i];
var scriptBuf = new Buffer(utxo.scriptPubKey, 'hex');
var scriptPubKey = new Script(scriptBuf);
var scriptType = scriptPubKey.classify();
Expand Down Expand Up @@ -582,16 +599,61 @@ TransactionBuilder.prototype._signMultiSig = function(walletKeyMap, input, txSig
};
};

var fnToSign = {};


TransactionBuilder.prototype._signScriptHash = function(walletKeyMap, input, txSigHash) {
var originalScriptBuf = this.tx.ins[input.i].s;


if (!this.hashToScriptMap)
throw new Error('hashToScriptMap not set');

var scriptHex = this.hashToScriptMap[input.address];
if (!scriptHex) return;

throw new Error('TX_SCRIPTHASH not supported yet');
};
var script = new Script(new Buffer(scriptHex,'hex'));
var scriptType = script.classify();
var scriptPubKeyHex = script.getBuffer().toString('hex');

if (!fnToSign[scriptType])
throw new Error('dont know how to sign p2sh script type'+ script.getRawOutType());

var newInput = {
address: 'TODO', // if p2pkubkeyhash -> get the address
i: input.i,
scriptPubKey: script,
scriptPubKeyHex: scriptPubKeyHex ,
scriptType: scriptType,
};

var txSigHash2 = this.tx.hashForSignature( script, input.i, this.signhash);
var ret = fnToSign[scriptType].call(this, walletKeyMap, newInput, txSigHash2);

var rc =1; //TODO : si alguno firmó...
if (ret.script) {

console.log('[TransactionBuilder.js.634] IN'); //TODO
var scriptSig = new Script(originalScriptBuf);
var len = scriptSig.chunks.length;
var scriptBufNotAlreadyAppended = scriptSig.chunks[len-1] !== undefined && (typeof scriptSig.chunks[len-1] == "number" || scriptSig.chunks[len-1].toString('hex') != scriptBuf.toString('hex'));
if (rc > 0 && scriptBufNotAlreadyAppended) {
scriptSig.chunks.push(scriptBuf);
scriptSig.updateBuffer();
ret.script = scriptSig.getBuffer();
}

if (scriptType == Script.TX_MULTISIG && scriptSig.finishedMultiSig())
{
scriptSig.removePlaceHolders();
scriptSig.prependOp0();
ret.script = scriptSig.getBuffer();
}
}

return ret;
};

var fnToSign = {};
fnToSign[Script.TX_PUBKEYHASH] = TransactionBuilder.prototype._signPubKeyHash;
fnToSign[Script.TX_PUBKEY] = TransactionBuilder.prototype._signPubKey;
fnToSign[Script.TX_MULTISIG] = TransactionBuilder.prototype._signMultiSig;
Expand All @@ -605,13 +667,13 @@ TransactionBuilder.prototype.sign = function(keys) {
walletKeyMap = TransactionBuilder._mapKeys(keys);



for (var i = 0; i < l; i++) {
var input = this.inputMap[i];

var txSigHash = this.tx.hashForSignature(
input.scriptPubKey, i, this.signhash);


var ret = fnToSign[input.scriptType].call(this, walletKeyMap, input, txSigHash);
if (ret && ret.script) {
tx.ins[i].s = ret.script; //esto no aqui TODO
Expand All @@ -624,6 +686,8 @@ TransactionBuilder.prototype.sign = function(keys) {
// [addr -> script]
TransactionBuilder.prototype.setHashToScriptMap = function(hashToScriptMap) {
this.hashToScriptMap= hashToScriptMap;

return this;
};


Expand Down
File renamed without changes.
File renamed without changes.
115 changes: 115 additions & 0 deletions examples/CreateAndSignTx-PayToScriptHash.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,115 @@
var run = function() {
bitcore = typeof (bitcore) === 'undefined' ? require('../bitcore') : bitcore;
var networks = require('../networks');
var WalletKey = bitcore.WalletKey;
var Script = bitcore.Script;
var Builder = bitcore.TransactionBuilder;
var opts = {network: networks.testnet};

console.log('## Network: ' + opts.network.name);

var input = {};
input.addr = "n2hoFVbPrYQf7RJwiRy1tkbuPPqyhAEfbp";
input.priv = "cS62Ej4SobZnpFQYN1PEEBr2KWf5sgRYYnELtumcG6WVCfxno39V";

// Complete with the corresponding UTXO you want to use
var utxos = [
{
address: "n2hoFVbPrYQf7RJwiRy1tkbuPPqyhAEfbp",
txid: "ff5c8b4912f6d056f0cf8431ec27032a73df22c167726267dd4cc0d7817a1e7d",
vout: 1,
ts: 1396290442,
scriptPubKey: "76a914e867aad8bd361f57c50adc37a0c018692b5b0c9a88ac",
amount: 0.5799,
confirmations: 7
}
];

var privs = [
"cP6JBHuQf7yqeqtdKRd22ibF3VehDv7G6BdzxSNABgrv3jFJUGoN",
"cQfRwF7XLSM5xGUpF8PZvob2MZyULvZPA2j5cat2RKDJrja7FtCZ",
"cUkYub4jtFVYymHh38yMMW36nJB4pXG5Pzd5QjResq79kAndkJcg",
"cMyBgowsyrJRufoKWob73rMQB1PBqDdwFt8z4TJ6APN2HkmX1Ttm",
"cN9yZCom6hAZpHtCp8ovE1zFa7RqDf3Cr4W6AwH2tp59Jjh9JcXu",
];

var pubkeys = []
privs.forEach(function(p) {
var wk = new WalletKey(opts);
wk.fromObj({priv: p});
pubkeys.push(bitcore.buffertools.toHex(wk.privKey.public));
});

// multisig p2sh
var opts = {nreq:3, pubkeys:pubkeys, amount:0.05};

// p2scriphash p2sh
//var opts = [{address: an_address, amount:0.05}];

var info = Builder.infoForP2sh(opts, 'testnet');
var p2shScript = info.scriptBufHex;
var p2shAddress = info.address;
var outs = [{address:p2shAddress, amount:0.05}];
var tx = new Builder(opts)
.setUnspent(utxos)
.setOutputs(outs)
.sign([input.priv])
.build();
var txHex = tx.serialize().toString('hex');
console.log('1) SEND TO P2SH TX: ', txHex);
console.log('[this example originally generated TXID: ba20653648a896ae95005b8f52847935a7313da06cd7295bb2cfc8b5c1b36c71 on testnet]\n\n\thttp://test.bitcore.io/tx/ba20653648a896ae95005b8f52847935a7313da06cd7295bb2cfc8b5c1b36c71\n\n');

//save scriptPubKey
var scriptPubKey = tx.outs[0].s.toString('hex');

/*
*
* REDDEEM TX
*/
var utxos2 = [
{
address: p2shAddress,
txid: "ba20653648a896ae95005b8f52847935a7313da06cd7295bb2cfc8b5c1b36c71",
vout: 0,
ts: 1396288753,
scriptPubKey: scriptPubKey,
amount: 0.05,
confirmations: 2
}
];

outs = [{address:input.addr, amount:0.04}];

var hashMap = {};
hashMap[p2shAddress]=p2shScript;

var b = new Builder(opts)
.setUnspent(utxos2)
.setHashToScriptMap(hashMap)
.setOutputs(outs)
.sign(privs);


tx= b.build();


var txHex = tx.serialize().toString('hex');
console.log('2) REDEEM SCRIPT: ', txHex);
console.log('=> Is signed status:', b.isFullySigned(), b.countInputMultiSig(0) );

console.log('[this example originally generated TXID: 2813c5a670d2c9d0527718f9d0ea896c78c3c8fc57b409e67308744fc7a7a98e on testnet]\n\n\thttp://test.bitcore.io/tx/2813c5a670d2c9d0527718f9d0ea896c78c3c8fc57b409e67308744fc7a7a98e');

};

// This is just for browser & mocha compatibility
if (typeof module !== 'undefined') {
module.exports.run = run;
if (require.main === module) {
run();
}
} else {
run();
}

////

0 comments on commit 4edab24

Please sign in to comment.