Skip to content

Commit 27897f2

Browse files
committed
Verify unsigned transaction will be accepted before transmitting signed transaction
1 parent 6800688 commit 27897f2

File tree

2 files changed

+128
-30
lines changed

2 files changed

+128
-30
lines changed

examples/bitcoinRpc.js

+64-28
Original file line numberDiff line numberDiff line change
@@ -181,10 +181,10 @@ async.waterfall([
181181
console.log('Bitcoind did not return a signed transaction');
182182
return cb(new Error('Missing signed tx'));
183183
}
184-
cb(null, signedTransaction.hex);
184+
cb(null, fundedRawTransaction, signedTransaction.hex);
185185
});
186186
},
187-
function displayTransactionToUserForApproval(signedRawTransaction, cb) {
187+
function getSignedTransactionSize(fundedRawTransaction, signedRawTransaction, cb) {
188188
let command = {
189189
jsonrpc: '1.0',
190190
method: 'decoderawtransaction',
@@ -197,52 +197,88 @@ async.waterfall([
197197
return cb(err);
198198
}
199199
if (!decodedTransaction) {
200-
console.log('Bitcoind did not return a decoded transaction');
200+
console.log('Bitcoind did not decode the transaction');
201201
return cb(new Error('Missing decoded tx'));
202202
}
203-
204-
console.log(JSON.stringify(decodedTransaction, null, 2));
205-
206-
promptly.confirm('Send payment shown above? (y/n)', function (err, accept) {
207-
if (!accept) {
208-
console.log('Payment cancelled');
209-
return cb(new Error('Payment Cancelled'));
210-
}
211-
return cb(null, signedRawTransaction);
212-
});
203+
cb(null, fundedRawTransaction, signedRawTransaction, decodedTransaction.vsize);
213204
});
214205
},
215-
function sendTransactionToServer(signedRawTransaction, cb) {
216-
paymentProtocol.sendPayment(config.currency, signedRawTransaction, paymentUrl, function (err, response) {
206+
function checkIfTransactionWillBeAccepted(fundedRawTransaction, signedRawTransaction, weightedSize, cb) {
207+
console.log('Sending unsigned transaction to server for verification...');
208+
209+
paymentProtocol.sendPaymentForVerification(config.currency, fundedRawTransaction, weightedSize, paymentUrl, (err) => {
217210
if (err) {
218-
console.log('Error sending payment to server', err);
211+
console.log('Error verifying payment with server', err);
219212
return cb(err);
220-
}
221-
else {
222-
console.log('Payment accepted by server');
213+
} else {
214+
console.log('Payment verified by server');
223215
return cb(null, signedRawTransaction);
224216
}
225217
});
226218
},
227-
//Note we only broadcast AFTER a SUCCESS response from the server
228-
function broadcastPayment(signedRawTransaction, cb) {
219+
//Note we only broadcast AFTER a SUCCESS response from the server verification request
220+
function displayTransactionToUserForApproval(signedRawTransaction, cb) {
229221
let command = {
230222
jsonrpc: '1.0',
231-
method: 'sendrawtransaction',
223+
method: 'decoderawtransaction',
232224
params: [signedRawTransaction]
233225
};
234226

235-
execRpcCommand(command, function (err, signedTransaction) {
227+
execRpcCommand(command, function (err, decodedTransaction) {
236228
if (err) {
237-
console.log('Error broadcasting transaction:', err);
229+
console.log('Error signing transaction:', err);
238230
return cb(err);
239231
}
240-
if (!signedTransaction) {
241-
console.log('Bitcoind failed to broadcast transaction');
242-
return cb(new Error('Failed to broadcast tx'));
232+
if (!decodedTransaction) {
233+
console.log('Bitcoind did not return a decoded transaction');
234+
return cb(new Error('Missing decoded tx'));
243235
}
244-
cb();
236+
237+
console.log(JSON.stringify(decodedTransaction, null, 2));
238+
239+
promptly.confirm('Send payment shown above? (y/n)', function (err, accept) {
240+
if (!accept) {
241+
console.log('Payment cancelled');
242+
return cb(new Error('Payment Cancelled'));
243+
}
244+
return cb(null, signedRawTransaction);
245+
});
245246
});
247+
},
248+
function broadcastPayment(signedRawTransaction, cb) {
249+
async.parallel([
250+
function sendToServer(cb) {
251+
paymentProtocol.broadcastPayment(config.currency, signedRawTransaction, paymentUrl, function(err) {
252+
if (err) {
253+
console.log('Error sending payment to server', err);
254+
return cb(err);
255+
}
256+
else {
257+
console.log('Payment accepted by server');
258+
return cb(null, signedRawTransaction);
259+
}
260+
});
261+
},
262+
function broadcastP2P(cb) {
263+
let command = {
264+
jsonrpc: '1.0',
265+
method: 'sendrawtransaction',
266+
params: [signedRawTransaction]
267+
};
268+
269+
execRpcCommand(command, function (err, signedTransaction) {
270+
if (err) {
271+
console.log('Error broadcasting transaction:', err);
272+
return cb(err);
273+
}
274+
if (!signedTransaction) {
275+
console.log('Bitcoind failed to broadcast transaction');
276+
return cb(new Error('Failed to broadcast tx'));
277+
}
278+
cb();
279+
});
280+
}
281+
], cb);
246282
}
247283
], function (err) {
248284
if (err) {

index.js

+64-2
Original file line numberDiff line numberDiff line change
@@ -195,11 +195,73 @@ PaymentProtocol.prototype.parsePaymentRequestAsync = util.promisify(PaymentProto
195195
/**
196196
* Sends a given payment to the server for validation
197197
* @param currency {string} Three letter currency code of proposed transaction (ie BTC, BCH)
198+
* @param unsignedRawTransaction {string} Hexadecimal format raw unsigned transaction
199+
* @param weightedSize {number} Weighted size of the transaction
200+
* @param url {string} the payment protocol specific url (https)
201+
* @param callback {function} (err, response)
202+
*/
203+
PaymentProtocol.prototype.sendPaymentForVerification = function sendPayment(currency, unsignedRawTransaction, weightedSize, url, callback) {
204+
let paymentResponse;
205+
206+
//Basic sanity checks
207+
if (typeof unsignedRawTransaction !== 'string') {
208+
return callback(new Error('unsignedRawTransaction must be a string'));
209+
}
210+
if (!/^[0-9a-f]+$/i.test(unsignedRawTransaction)) {
211+
return callback(new Error('unsignedRawTransaction must be in hexadecimal format'));
212+
}
213+
if (typeof weightedSize !== 'number' || parseInt(weightedSize) !== weightedSize) {
214+
return callback(new Error('weightedSize must be an integer'));
215+
}
216+
217+
let requestOptions = _.merge(this.options, {
218+
url: url,
219+
headers: {
220+
'Content-Type': 'application/verify-payment'
221+
},
222+
body: JSON.stringify({
223+
currency: currency,
224+
transactions: [unsignedRawTransaction],
225+
weightedSize: weightedSize
226+
})
227+
});
228+
229+
request.post(requestOptions, (err, response) => {
230+
if (err) {
231+
return callback(err);
232+
}
233+
if (response.statusCode !== 200) {
234+
return callback(new Error(response.body.toString()));
235+
}
236+
237+
try {
238+
paymentResponse = JSON.parse(response.body);
239+
}
240+
catch (e) {
241+
return callback(new Error('Unable to parse response from server'));
242+
}
243+
244+
callback(null, paymentResponse);
245+
});
246+
};
247+
248+
/**
249+
* Sends a given payment to the server for validation
250+
* @param currency {string} Three letter currency code of proposed transaction (ie BTC, BCH)
251+
* @param unsignedRawTransaction {string} Hexadecimal format raw unsigned transaction
252+
* @param weightedSize {number} Weighted size of the transaction
253+
* @param url {string} the payment protocol specific url (https)
254+
*/
255+
PaymentProtocol.prototype.sendPaymentForVerificationAsync = util.promisify(PaymentProtocol.prototype.sendPaymentForVerification);
256+
257+
/**
258+
* Sends actual payment to server
259+
* @param currency {string} Three letter currency code of proposed transaction (ie BTC, BCH)
198260
* @param signedRawTransaction {string} Hexadecimal format raw signed transaction
199261
* @param url {string} the payment protocol specific url (https)
200262
* @param callback {function} (err, response)
201263
*/
202-
PaymentProtocol.prototype.sendPayment = function sendPayment(currency, signedRawTransaction, url, callback) {
264+
PaymentProtocol.prototype.broadcastPayment = function broadcastPayment(currency, signedRawTransaction, url, callback) {
203265
let paymentResponse;
204266

205267
//Basic sanity checks
@@ -246,7 +308,7 @@ PaymentProtocol.prototype.sendPayment = function sendPayment(currency, signedRaw
246308
* @param signedRawTransaction {string} Hexadecimal format raw signed transaction
247309
* @param url {string} the payment protocol specific url (https)
248310
*/
249-
PaymentProtocol.prototype.sendPaymentAsync = util.promisify(PaymentProtocol.prototype.sendPayment);
311+
PaymentProtocol.prototype.broadcastPaymentAsync = util.promisify(PaymentProtocol.prototype.broadcastPayment);
250312

251313
module.exports = PaymentProtocol;
252314

0 commit comments

Comments
 (0)