@bitgo-beta/sdk-api 1.10.1-beta.58 → 1.10.1-beta.580
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +751 -0
- package/dist/package.json +19 -24
- package/dist/src/api.d.ts +1 -1
- package/dist/src/api.d.ts.map +1 -1
- package/dist/src/api.js +23 -20
- package/dist/src/bitgoAPI.d.ts +43 -5
- package/dist/src/bitgoAPI.d.ts.map +1 -1
- package/dist/src/bitgoAPI.js +255 -54
- package/dist/src/encrypt.d.ts +5 -3
- package/dist/src/encrypt.d.ts.map +1 -1
- package/dist/src/encrypt.js +31 -3
- package/dist/src/index.js +6 -2
- package/dist/src/types.d.ts +28 -2
- package/dist/src/types.d.ts.map +1 -1
- package/dist/src/types.js +2 -2
- package/dist/src/util.js +25 -2
- package/dist/src/v1/blockchain.js +14 -11
- package/dist/src/v1/keychains.js +15 -12
- package/dist/src/v1/markets.js +8 -5
- package/dist/src/v1/pendingapproval.js +43 -17
- package/dist/src/v1/pendingapprovals.js +31 -5
- package/dist/src/v1/signPsbt.d.ts +14 -0
- package/dist/src/v1/signPsbt.d.ts.map +1 -0
- package/dist/src/v1/signPsbt.js +60 -0
- package/dist/src/v1/transactionBuilder.js +133 -91
- package/dist/src/v1/travelRule.js +48 -22
- package/dist/src/v1/verifyAddress.d.ts +6 -0
- package/dist/src/v1/verifyAddress.d.ts.map +1 -0
- package/dist/src/v1/verifyAddress.js +41 -0
- package/dist/src/v1/wallet.d.ts.map +1 -1
- package/dist/src/v1/wallet.js +307 -167
- package/dist/src/v1/wallets.js +53 -27
- package/package.json +19 -24
- package/dist/web/main.js +0 -2
- package/dist/web/main.js.LICENSE.txt +0 -103
|
@@ -2,6 +2,32 @@
|
|
|
2
2
|
/**
|
|
3
3
|
* @hidden
|
|
4
4
|
*/
|
|
5
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
6
|
+
if (k2 === undefined) k2 = k;
|
|
7
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
8
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
9
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
10
|
+
}
|
|
11
|
+
Object.defineProperty(o, k2, desc);
|
|
12
|
+
}) : (function(o, m, k, k2) {
|
|
13
|
+
if (k2 === undefined) k2 = k;
|
|
14
|
+
o[k2] = m[k];
|
|
15
|
+
}));
|
|
16
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
17
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
18
|
+
}) : function(o, v) {
|
|
19
|
+
o["default"] = v;
|
|
20
|
+
});
|
|
21
|
+
var __importStar = (this && this.__importStar) || function (mod) {
|
|
22
|
+
if (mod && mod.__esModule) return mod;
|
|
23
|
+
var result = {};
|
|
24
|
+
if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
|
|
25
|
+
__setModuleDefault(result, mod);
|
|
26
|
+
return result;
|
|
27
|
+
};
|
|
28
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
29
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
30
|
+
};
|
|
5
31
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
32
|
/**
|
|
7
33
|
*/
|
|
@@ -12,13 +38,14 @@ Object.defineProperty(exports, "__esModule", { value: true });
|
|
|
12
38
|
// Copyright 2014, BitGo, Inc. All Rights Reserved.
|
|
13
39
|
//
|
|
14
40
|
const utxo_lib_1 = require("@bitgo-beta/utxo-lib");
|
|
15
|
-
const
|
|
16
|
-
const utxolib = require("@bitgo-beta/utxo-lib");
|
|
17
|
-
const
|
|
41
|
+
const bluebird_1 = __importDefault(require("bluebird"));
|
|
42
|
+
const utxolib = __importStar(require("@bitgo-beta/utxo-lib"));
|
|
43
|
+
const lodash_1 = __importDefault(require("lodash"));
|
|
18
44
|
const unspents_1 = require("@bitgo-beta/unspents");
|
|
19
45
|
const debugLib = require("debug");
|
|
20
46
|
const debug = debugLib('bitgo:v1:txb');
|
|
21
47
|
const sdk_core_1 = require("@bitgo-beta/sdk-core");
|
|
48
|
+
const verifyAddress_1 = require("./verifyAddress");
|
|
22
49
|
//
|
|
23
50
|
// TransactionBuilder
|
|
24
51
|
// @params:
|
|
@@ -38,6 +65,7 @@ const sdk_core_1 = require("@bitgo-beta/sdk-core");
|
|
|
38
65
|
// feeSingleKeySourceAddress: Use this single key address to pay fees
|
|
39
66
|
// feeSingleKeyWIF: Use the address based on this private key to pay fees
|
|
40
67
|
// unspentsFetchParams: Extra parameters to use for fetching unspents for this transaction
|
|
68
|
+
// unspents: array of unspent objects to use while constructing the transaction instead of fetching from the API
|
|
41
69
|
exports.createTransaction = function (params) {
|
|
42
70
|
const minConfirms = params.minConfirms || 0;
|
|
43
71
|
const validate = params.validate === undefined ? true : params.validate;
|
|
@@ -47,29 +75,29 @@ exports.createTransaction = function (params) {
|
|
|
47
75
|
let estTxSize;
|
|
48
76
|
let travelInfos;
|
|
49
77
|
// Sanity check the arguments passed in
|
|
50
|
-
if (!
|
|
51
|
-
(params.fee && !
|
|
52
|
-
(params.feeRate && !
|
|
53
|
-
!
|
|
54
|
-
(params.forceChangeAtEnd && !
|
|
55
|
-
(params.changeAddress && !
|
|
56
|
-
(params.noSplitChange && !
|
|
57
|
-
(params.targetWalletUnspents && !
|
|
58
|
-
(validate && !
|
|
59
|
-
(params.enforceMinConfirmsForChange && !
|
|
60
|
-
(params.minUnspentSize && !
|
|
61
|
-
(params.maxFeeRate && !
|
|
78
|
+
if (!lodash_1.default.isObject(params.wallet) ||
|
|
79
|
+
(params.fee && !lodash_1.default.isNumber(params.fee)) ||
|
|
80
|
+
(params.feeRate && !lodash_1.default.isNumber(params.feeRate)) ||
|
|
81
|
+
!lodash_1.default.isInteger(minConfirms) ||
|
|
82
|
+
(params.forceChangeAtEnd && !lodash_1.default.isBoolean(params.forceChangeAtEnd)) ||
|
|
83
|
+
(params.changeAddress && !lodash_1.default.isString(params.changeAddress)) ||
|
|
84
|
+
(params.noSplitChange && !lodash_1.default.isBoolean(params.noSplitChange)) ||
|
|
85
|
+
(params.targetWalletUnspents && !lodash_1.default.isInteger(params.targetWalletUnspents)) ||
|
|
86
|
+
(validate && !lodash_1.default.isBoolean(validate)) ||
|
|
87
|
+
(params.enforceMinConfirmsForChange && !lodash_1.default.isBoolean(params.enforceMinConfirmsForChange)) ||
|
|
88
|
+
(params.minUnspentSize && !lodash_1.default.isNumber(params.minUnspentSize)) ||
|
|
89
|
+
(params.maxFeeRate && !lodash_1.default.isNumber(params.maxFeeRate)) ||
|
|
62
90
|
// this should be an array and its length must be at least 1
|
|
63
91
|
(params.unspents && (!Array.isArray(params.unspents) || params.unspents.length < 1)) ||
|
|
64
|
-
(params.feeTxConfirmTarget && !
|
|
65
|
-
(params.instant && !
|
|
66
|
-
(params.bitgoFee && !
|
|
67
|
-
(params.unspentsFetchParams && !
|
|
92
|
+
(params.feeTxConfirmTarget && !lodash_1.default.isInteger(params.feeTxConfirmTarget)) ||
|
|
93
|
+
(params.instant && !lodash_1.default.isBoolean(params.instant)) ||
|
|
94
|
+
(params.bitgoFee && !lodash_1.default.isObject(params.bitgoFee)) ||
|
|
95
|
+
(params.unspentsFetchParams && !lodash_1.default.isObject(params.unspentsFetchParams))) {
|
|
68
96
|
throw new Error('invalid argument');
|
|
69
97
|
}
|
|
70
98
|
const bitgo = params.wallet.bitgo;
|
|
71
99
|
const constants = bitgo.getConstants();
|
|
72
|
-
const network = sdk_core_1.getNetwork(sdk_core_1.common.Environments[bitgo.getEnv()].network);
|
|
100
|
+
const network = (0, sdk_core_1.getNetwork)(sdk_core_1.common.Environments[bitgo.getEnv()].network);
|
|
73
101
|
// The user can specify a seperate, single-key wallet for the purposes of paying miner's fees
|
|
74
102
|
// When creating a transaction this can be specified as an input address or the private key in WIF
|
|
75
103
|
let feeSingleKeySourceAddress;
|
|
@@ -85,7 +113,7 @@ exports.createTransaction = function (params) {
|
|
|
85
113
|
}
|
|
86
114
|
if (params.feeSingleKeyWIF) {
|
|
87
115
|
const feeSingleKey = utxolib.ECPair.fromWIF(params.feeSingleKeyWIF, network);
|
|
88
|
-
feeSingleKeySourceAddress = sdk_core_1.getAddressP2PKH(feeSingleKey);
|
|
116
|
+
feeSingleKeySourceAddress = (0, sdk_core_1.getAddressP2PKH)(feeSingleKey);
|
|
89
117
|
// If the user specifies both, check to make sure the feeSingleKeySourceAddress corresponds to the address of feeSingleKeyWIF
|
|
90
118
|
if (params.feeSingleKeySourceAddress && params.feeSingleKeySourceAddress !== feeSingleKeySourceAddress) {
|
|
91
119
|
throw new Error('feeSingleKeySourceAddress: ' +
|
|
@@ -94,23 +122,23 @@ exports.createTransaction = function (params) {
|
|
|
94
122
|
feeSingleKeySourceAddress);
|
|
95
123
|
}
|
|
96
124
|
}
|
|
97
|
-
if (!
|
|
125
|
+
if (!lodash_1.default.isObject(params.recipients)) {
|
|
98
126
|
throw new Error('recipients must be array of { address: abc, amount: 100000 } objects');
|
|
99
127
|
}
|
|
100
128
|
let feeParamsDefined = 0;
|
|
101
|
-
if (!
|
|
129
|
+
if (!lodash_1.default.isUndefined(params.fee)) {
|
|
102
130
|
feeParamsDefined++;
|
|
103
131
|
}
|
|
104
|
-
if (!
|
|
132
|
+
if (!lodash_1.default.isUndefined(params.feeRate)) {
|
|
105
133
|
feeParamsDefined++;
|
|
106
134
|
}
|
|
107
|
-
if (!
|
|
135
|
+
if (!lodash_1.default.isUndefined(params.feeTxConfirmTarget)) {
|
|
108
136
|
feeParamsDefined++;
|
|
109
137
|
}
|
|
110
138
|
if (feeParamsDefined > 1) {
|
|
111
139
|
throw new Error('cannot specify more than one of fee, feeRate and feeTxConfirmTarget');
|
|
112
140
|
}
|
|
113
|
-
if (
|
|
141
|
+
if (lodash_1.default.isUndefined(params.maxFeeRate)) {
|
|
114
142
|
params.maxFeeRate = constants.maxFeeRate;
|
|
115
143
|
}
|
|
116
144
|
// Convert the old format of params.recipients (dictionary of address:amount) to new format: { destinationAddress, amount }
|
|
@@ -142,14 +170,11 @@ exports.createTransaction = function (params) {
|
|
|
142
170
|
let fee = params.fee;
|
|
143
171
|
let feeRate = params.feeRate;
|
|
144
172
|
// Flag indicating whether this class will compute the fee
|
|
145
|
-
const shouldComputeBestFee =
|
|
173
|
+
const shouldComputeBestFee = lodash_1.default.isUndefined(fee);
|
|
146
174
|
let totalOutputAmount = 0;
|
|
147
175
|
recipients.forEach(function (recipient) {
|
|
148
|
-
if (
|
|
149
|
-
|
|
150
|
-
utxolib.address.fromBase58Check(recipient.address, network);
|
|
151
|
-
}
|
|
152
|
-
catch (e) {
|
|
176
|
+
if (lodash_1.default.isString(recipient.address)) {
|
|
177
|
+
if (!(0, verifyAddress_1.verifyAddress)(recipient.address, network)) {
|
|
153
178
|
throw new Error('invalid bitcoin address: ' + recipient.address);
|
|
154
179
|
}
|
|
155
180
|
if (!!recipient.script) {
|
|
@@ -159,7 +184,7 @@ exports.createTransaction = function (params) {
|
|
|
159
184
|
}
|
|
160
185
|
}
|
|
161
186
|
}
|
|
162
|
-
if (!
|
|
187
|
+
if (!lodash_1.default.isInteger(recipient.amount) || recipient.amount < 0) {
|
|
163
188
|
throw new Error('invalid amount for ' + recipient.address + ': ' + recipient.amount);
|
|
164
189
|
}
|
|
165
190
|
totalOutputAmount += recipient.amount;
|
|
@@ -168,7 +193,7 @@ exports.createTransaction = function (params) {
|
|
|
168
193
|
totalOutputAmount += opReturn.amount;
|
|
169
194
|
});
|
|
170
195
|
let bitgoFeeInfo = params.bitgoFee;
|
|
171
|
-
if (bitgoFeeInfo && (!
|
|
196
|
+
if (bitgoFeeInfo && (!lodash_1.default.isInteger(bitgoFeeInfo.amount) || !lodash_1.default.isString(bitgoFeeInfo.address))) {
|
|
172
197
|
throw new Error('invalid bitgoFeeInfo');
|
|
173
198
|
}
|
|
174
199
|
// The total amount needed for this transaction.
|
|
@@ -184,10 +209,11 @@ exports.createTransaction = function (params) {
|
|
|
184
209
|
// The sum of the input values for this transaction.
|
|
185
210
|
let inputAmount;
|
|
186
211
|
let changeOutputs = [];
|
|
212
|
+
let containsUncompressedPublicKeys = false;
|
|
187
213
|
// The transaction.
|
|
188
214
|
let transaction = utxolib.bitgo.createTransactionBuilderForNetwork(network);
|
|
189
215
|
const getBitGoFee = function () {
|
|
190
|
-
return
|
|
216
|
+
return bluebird_1.default.try(function () {
|
|
191
217
|
if (bitgoFeeInfo) {
|
|
192
218
|
return;
|
|
193
219
|
}
|
|
@@ -205,7 +231,7 @@ exports.createTransaction = function (params) {
|
|
|
205
231
|
});
|
|
206
232
|
};
|
|
207
233
|
const getBitGoFeeAddress = function () {
|
|
208
|
-
return
|
|
234
|
+
return bluebird_1.default.try(function () {
|
|
209
235
|
// If we don't have bitgoFeeInfo, or address is already set, don't get a new one
|
|
210
236
|
if (!bitgoFeeInfo || bitgoFeeInfo.address) {
|
|
211
237
|
return;
|
|
@@ -253,14 +279,14 @@ exports.createTransaction = function (params) {
|
|
|
253
279
|
})
|
|
254
280
|
.catch(function (e) {
|
|
255
281
|
// sanity check failed on tx size
|
|
256
|
-
if (
|
|
257
|
-
return
|
|
282
|
+
if (lodash_1.default.includes(e.message, 'invalid txSize')) {
|
|
283
|
+
return bluebird_1.default.reject(e);
|
|
258
284
|
}
|
|
259
285
|
else {
|
|
260
286
|
// couldn't estimate the fee, proceed using the default
|
|
261
287
|
feeRate = constants.fallbackFeeRate;
|
|
262
288
|
console.log('Error estimating fee for send from ' + params.wallet.id() + ': ' + e.message);
|
|
263
|
-
return
|
|
289
|
+
return bluebird_1.default.resolve();
|
|
264
290
|
}
|
|
265
291
|
});
|
|
266
292
|
}
|
|
@@ -273,7 +299,7 @@ exports.createTransaction = function (params) {
|
|
|
273
299
|
return;
|
|
274
300
|
}
|
|
275
301
|
// Get enough unspents for the requested amount
|
|
276
|
-
const options =
|
|
302
|
+
const options = lodash_1.default.merge({}, params.unspentsFetchParams || {}, {
|
|
277
303
|
target: totalAmount,
|
|
278
304
|
minSize: params.minUnspentSize || 0,
|
|
279
305
|
instant: params.instant,
|
|
@@ -283,6 +309,7 @@ exports.createTransaction = function (params) {
|
|
|
283
309
|
options.instant = params.instant; // insist on instant unspents only
|
|
284
310
|
}
|
|
285
311
|
return params.wallet.unspentsPaged(options).then(function (results) {
|
|
312
|
+
console.log(`Unspents fetched\n: ${JSON.stringify(results, null, 2)}`);
|
|
286
313
|
totalUnspentsCount = results.total;
|
|
287
314
|
fetchedUnspentsCount = results.count;
|
|
288
315
|
unspents = results.unspents.filter(function (u) {
|
|
@@ -297,7 +324,7 @@ exports.createTransaction = function (params) {
|
|
|
297
324
|
throw Error('0 unspents available for transaction creation');
|
|
298
325
|
}
|
|
299
326
|
// create array of unconfirmed unspent ID strings of the form "txHash:outputIndex"
|
|
300
|
-
zeroConfUnspentTxIds =
|
|
327
|
+
zeroConfUnspentTxIds = (0, lodash_1.default)(results.unspents)
|
|
301
328
|
.filter(function (u) {
|
|
302
329
|
return !u.confirmations;
|
|
303
330
|
})
|
|
@@ -305,7 +332,7 @@ exports.createTransaction = function (params) {
|
|
|
305
332
|
return u.tx_hash + ':' + u.tx_output_n;
|
|
306
333
|
})
|
|
307
334
|
.value();
|
|
308
|
-
if (
|
|
335
|
+
if (lodash_1.default.isEmpty(zeroConfUnspentTxIds)) {
|
|
309
336
|
// we don't want to pass an empty array of inputs to the server, because it assumes if the
|
|
310
337
|
// inputs arguments exists, it contains values
|
|
311
338
|
zeroConfUnspentTxIds = undefined;
|
|
@@ -345,9 +372,9 @@ exports.createTransaction = function (params) {
|
|
|
345
372
|
}
|
|
346
373
|
inputAmount = 0;
|
|
347
374
|
// Calculate the cost of spending a single input, i.e. the smallest economical unspent value
|
|
348
|
-
return
|
|
349
|
-
if (
|
|
350
|
-
return !
|
|
375
|
+
return bluebird_1.default.try(function () {
|
|
376
|
+
if (lodash_1.default.isNumber(params.feeRate) || lodash_1.default.isNumber(params.originalFeeRate)) {
|
|
377
|
+
return !lodash_1.default.isUndefined(params.feeRate) ? params.feeRate : params.originalFeeRate;
|
|
351
378
|
}
|
|
352
379
|
else {
|
|
353
380
|
return bitgo
|
|
@@ -363,12 +390,12 @@ exports.createTransaction = function (params) {
|
|
|
363
390
|
.then(function (feeRate) {
|
|
364
391
|
// Don't spend inputs that cannot pay for their own cost.
|
|
365
392
|
let minInputValue = 0;
|
|
366
|
-
if (
|
|
393
|
+
if (lodash_1.default.isInteger(params.minUnspentSize)) {
|
|
367
394
|
minInputValue = params.minUnspentSize;
|
|
368
395
|
}
|
|
369
396
|
let prunedUnspentCount = 0;
|
|
370
397
|
const originalUnspentCount = unspents.length;
|
|
371
|
-
unspents =
|
|
398
|
+
unspents = lodash_1.default.filter(unspents, function (unspent) {
|
|
372
399
|
const isSegwitInput = !!unspent.witnessScript;
|
|
373
400
|
const currentInputSize = isSegwitInput ? unspents_1.VirtualSizes.txP2shP2wshInputSize : unspents_1.VirtualSizes.txP2shInputSize;
|
|
374
401
|
const feeBasedMinInputValue = (feeRate * currentInputSize) / 1000;
|
|
@@ -383,14 +410,14 @@ exports.createTransaction = function (params) {
|
|
|
383
410
|
inputSize: currentInputSize,
|
|
384
411
|
unspent: unspent,
|
|
385
412
|
};
|
|
386
|
-
|
|
413
|
+
debug(`pruning unspent: ${JSON.stringify(pruneDetails, null, 4)}`);
|
|
387
414
|
prunedUnspentCount++;
|
|
388
415
|
return false;
|
|
389
416
|
}
|
|
390
417
|
return true;
|
|
391
418
|
});
|
|
392
419
|
if (prunedUnspentCount > 0) {
|
|
393
|
-
|
|
420
|
+
debug(`pruned ${prunedUnspentCount} out of ${originalUnspentCount} unspents`);
|
|
394
421
|
}
|
|
395
422
|
if (unspents.length === 0) {
|
|
396
423
|
throw new Error('insufficient funds');
|
|
@@ -429,7 +456,11 @@ exports.createTransaction = function (params) {
|
|
|
429
456
|
(bitgoFeeInfo && bitgoFeeInfo.amount > 0 ? 1 : 0) + // add output for bitgo fee
|
|
430
457
|
(feeSingleKeySourceAddress ? 1 : 0),
|
|
431
458
|
};
|
|
459
|
+
// As per the response of get unspents API, for v1 safe wallets redeemScript is returned
|
|
460
|
+
// in the response in hex format
|
|
461
|
+
containsUncompressedPublicKeys = unspents.some((u) => u.redeemScript.length === 201 * 2 /* hex length is twice the length in bytes */);
|
|
432
462
|
estTxSize = estimateTransactionSize({
|
|
463
|
+
containsUncompressedPublicKeys,
|
|
433
464
|
nP2shInputs: txInfo.nP2shInputs,
|
|
434
465
|
nP2shP2wshInputs: txInfo.nP2shP2wshInputs,
|
|
435
466
|
nP2pkhInputs: txInfo.nP2pkhInputs,
|
|
@@ -440,6 +471,7 @@ exports.createTransaction = function (params) {
|
|
|
440
471
|
.then(function () {
|
|
441
472
|
minerFeeInfo = exports.calculateMinerFeeInfo({
|
|
442
473
|
bitgo: params.wallet.bitgo,
|
|
474
|
+
containsUncompressedPublicKeys,
|
|
443
475
|
feeRate: feeRate,
|
|
444
476
|
nP2shInputs: txInfo.nP2shInputs,
|
|
445
477
|
nP2shP2wshInputs: txInfo.nP2shP2wshInputs,
|
|
@@ -448,7 +480,7 @@ exports.createTransaction = function (params) {
|
|
|
448
480
|
});
|
|
449
481
|
if (shouldComputeBestFee) {
|
|
450
482
|
const approximateFee = minerFeeInfo.fee;
|
|
451
|
-
const shouldRecurse =
|
|
483
|
+
const shouldRecurse = lodash_1.default.isUndefined(fee) || approximateFee > fee;
|
|
452
484
|
fee = approximateFee;
|
|
453
485
|
// Recompute totalAmount from scratch
|
|
454
486
|
totalAmount = fee + totalOutputAmount;
|
|
@@ -464,7 +496,7 @@ exports.createTransaction = function (params) {
|
|
|
464
496
|
}
|
|
465
497
|
const totalFee = fee + (bitgoFeeInfo ? bitgoFeeInfo.amount : 0);
|
|
466
498
|
if (feeSingleKeySourceAddress) {
|
|
467
|
-
const summedSingleKeyUnspents =
|
|
499
|
+
const summedSingleKeyUnspents = lodash_1.default.sumBy(feeSingleKeyUnspents, 'value');
|
|
468
500
|
if (totalFee > summedSingleKeyUnspents) {
|
|
469
501
|
const err = new Error('Insufficient fee amount available in single key fee source: ' + summedSingleKeyUnspents);
|
|
470
502
|
err.result = {
|
|
@@ -475,7 +507,7 @@ exports.createTransaction = function (params) {
|
|
|
475
507
|
bitgoFee: bitgoFeeInfo,
|
|
476
508
|
txInfo: txInfo,
|
|
477
509
|
};
|
|
478
|
-
return
|
|
510
|
+
return bluebird_1.default.reject(err);
|
|
479
511
|
}
|
|
480
512
|
}
|
|
481
513
|
if (inputAmount < (feeSingleKeySourceAddress ? totalOutputAmount : totalAmount)) {
|
|
@@ -503,7 +535,7 @@ exports.createTransaction = function (params) {
|
|
|
503
535
|
bitgoFee: bitgoFeeInfo,
|
|
504
536
|
txInfo: txInfo,
|
|
505
537
|
};
|
|
506
|
-
return
|
|
538
|
+
return bluebird_1.default.reject(err);
|
|
507
539
|
}
|
|
508
540
|
});
|
|
509
541
|
};
|
|
@@ -515,10 +547,10 @@ exports.createTransaction = function (params) {
|
|
|
515
547
|
const outputs = [];
|
|
516
548
|
recipients.forEach(function (recipient) {
|
|
517
549
|
let script;
|
|
518
|
-
if (
|
|
550
|
+
if (lodash_1.default.isString(recipient.address)) {
|
|
519
551
|
script = utxolib.address.toOutputScript(recipient.address, network);
|
|
520
552
|
}
|
|
521
|
-
else if (
|
|
553
|
+
else if (lodash_1.default.isObject(recipient.script)) {
|
|
522
554
|
script = recipient.script;
|
|
523
555
|
}
|
|
524
556
|
else {
|
|
@@ -526,7 +558,7 @@ exports.createTransaction = function (params) {
|
|
|
526
558
|
}
|
|
527
559
|
// validate travelInfo if it exists
|
|
528
560
|
let travelInfo;
|
|
529
|
-
if (!
|
|
561
|
+
if (!lodash_1.default.isEmpty(recipient.travelInfo)) {
|
|
530
562
|
travelInfo = recipient.travelInfo;
|
|
531
563
|
// Better to avoid trouble now, before tx is created
|
|
532
564
|
bitgo.travelRule().validateTravelInfo(travelInfo);
|
|
@@ -564,7 +596,7 @@ exports.createTransaction = function (params) {
|
|
|
564
596
|
return result;
|
|
565
597
|
});
|
|
566
598
|
}
|
|
567
|
-
let extraChangeTotal =
|
|
599
|
+
let extraChangeTotal = lodash_1.default.sum(extraChangeAmounts);
|
|
568
600
|
// Sanity check
|
|
569
601
|
if (extraChangeTotal > changeAmount) {
|
|
570
602
|
extraChangeAmounts = [];
|
|
@@ -579,7 +611,7 @@ exports.createTransaction = function (params) {
|
|
|
579
611
|
if (!thisAmount) {
|
|
580
612
|
return result;
|
|
581
613
|
}
|
|
582
|
-
return
|
|
614
|
+
return bluebird_1.default.try(function () {
|
|
583
615
|
if (params.changeAddress) {
|
|
584
616
|
// If user passed a change address, use it for all outputs
|
|
585
617
|
return params.changeAddress;
|
|
@@ -600,7 +632,7 @@ exports.createTransaction = function (params) {
|
|
|
600
632
|
return addChangeOutputs();
|
|
601
633
|
};
|
|
602
634
|
// Add change output(s) and instant fee output if applicable
|
|
603
|
-
return
|
|
635
|
+
return bluebird_1.default.try(function () {
|
|
604
636
|
return getChangeOutputs(inputAmount - totalAmount);
|
|
605
637
|
}).then(function (result) {
|
|
606
638
|
changeOutputs = result;
|
|
@@ -613,14 +645,14 @@ exports.createTransaction = function (params) {
|
|
|
613
645
|
output.script = utxolib.address.toOutputScript(output.address, network);
|
|
614
646
|
}
|
|
615
647
|
// decide where to put the outputs - default is to randomize unless forced to end
|
|
616
|
-
const outputIndex = params.forceChangeAtEnd ? outputs.length :
|
|
648
|
+
const outputIndex = params.forceChangeAtEnd ? outputs.length : lodash_1.default.random(0, outputs.length);
|
|
617
649
|
outputs.splice(outputIndex, 0, output);
|
|
618
650
|
});
|
|
619
651
|
// Add all outputs to the transaction
|
|
620
652
|
outputs.forEach(function (output) {
|
|
621
653
|
transaction.addOutput(output.script, output.amount);
|
|
622
654
|
});
|
|
623
|
-
travelInfos =
|
|
655
|
+
travelInfos = (0, lodash_1.default)(outputs)
|
|
624
656
|
.map(function (output, index) {
|
|
625
657
|
const result = output.travelInfo;
|
|
626
658
|
if (!result) {
|
|
@@ -636,11 +668,11 @@ exports.createTransaction = function (params) {
|
|
|
636
668
|
// Serialize the transaction, returning what is needed to sign it
|
|
637
669
|
const serialize = function () {
|
|
638
670
|
// only need to return the unspents that were used and just the chainPath, redeemScript, and instant flag
|
|
639
|
-
const pickedUnspents =
|
|
640
|
-
return
|
|
671
|
+
const pickedUnspents = lodash_1.default.map(unspents, function (unspent) {
|
|
672
|
+
return lodash_1.default.pick(unspent, ['chainPath', 'redeemScript', 'instant', 'witnessScript', 'script', 'value']);
|
|
641
673
|
});
|
|
642
|
-
const prunedUnspents =
|
|
643
|
-
|
|
674
|
+
const prunedUnspents = lodash_1.default.slice(pickedUnspents, 0, transaction.tx.ins.length - feeSingleKeyUnspentsUsed.length);
|
|
675
|
+
lodash_1.default.each(feeSingleKeyUnspentsUsed, function (feeUnspent) {
|
|
644
676
|
prunedUnspents.push({ redeemScript: false, chainPath: false }); // mark as false to signify a non-multisig address
|
|
645
677
|
});
|
|
646
678
|
const result = {
|
|
@@ -648,7 +680,7 @@ exports.createTransaction = function (params) {
|
|
|
648
680
|
unspents: prunedUnspents,
|
|
649
681
|
fee: fee,
|
|
650
682
|
changeAddresses: changeOutputs.map(function (co) {
|
|
651
|
-
return
|
|
683
|
+
return lodash_1.default.pick(co, ['address', 'path', 'amount']);
|
|
652
684
|
}),
|
|
653
685
|
walletId: params.wallet.id(),
|
|
654
686
|
walletKeychains: params.wallet.keychains,
|
|
@@ -661,15 +693,15 @@ exports.createTransaction = function (params) {
|
|
|
661
693
|
};
|
|
662
694
|
// Add for backwards compatibility
|
|
663
695
|
if (result.instant && bitgoFeeInfo) {
|
|
664
|
-
result.instantFee =
|
|
696
|
+
result.instantFee = lodash_1.default.pick(bitgoFeeInfo, ['amount', 'address']);
|
|
665
697
|
}
|
|
666
698
|
return result;
|
|
667
699
|
};
|
|
668
|
-
return
|
|
700
|
+
return bluebird_1.default.try(function () {
|
|
669
701
|
return getBitGoFee();
|
|
670
702
|
})
|
|
671
703
|
.then(function () {
|
|
672
|
-
return
|
|
704
|
+
return bluebird_1.default.all([getBitGoFeeAddress(), getUnspents(), getUnspentsForSingleKey()]);
|
|
673
705
|
})
|
|
674
706
|
.then(collectInputs)
|
|
675
707
|
.then(collectOutputs)
|
|
@@ -687,29 +719,37 @@ exports.createTransaction = function (params) {
|
|
|
687
719
|
* @returns size: estimated size of the transaction in bytes
|
|
688
720
|
*/
|
|
689
721
|
const estimateTransactionSize = function (params) {
|
|
690
|
-
if (!
|
|
722
|
+
if (!lodash_1.default.isInteger(params.nP2shInputs) || params.nP2shInputs < 0) {
|
|
691
723
|
throw new Error('expecting positive nP2shInputs');
|
|
692
724
|
}
|
|
693
|
-
if (!
|
|
725
|
+
if (!lodash_1.default.isInteger(params.nP2pkhInputs) || params.nP2pkhInputs < 0) {
|
|
694
726
|
throw new Error('expecting positive nP2pkhInputs to be numeric');
|
|
695
727
|
}
|
|
696
|
-
if (!
|
|
728
|
+
if (!lodash_1.default.isInteger(params.nP2shP2wshInputs) || params.nP2shP2wshInputs < 0) {
|
|
697
729
|
throw new Error('expecting positive nP2shP2wshInputs to be numeric');
|
|
698
730
|
}
|
|
699
731
|
if (params.nP2shInputs + params.nP2shP2wshInputs < 1) {
|
|
700
732
|
throw new Error('expecting at least one nP2shInputs or nP2shP2wshInputs');
|
|
701
733
|
}
|
|
702
|
-
if (!
|
|
734
|
+
if (!lodash_1.default.isInteger(params.nOutputs) || params.nOutputs < 1) {
|
|
703
735
|
throw new Error('expecting positive nOutputs');
|
|
704
736
|
}
|
|
705
|
-
|
|
737
|
+
// The size of an uncompressed public key is 32 bytes more than the compressed key,
|
|
738
|
+
// and hence, needs to be accounted for in the transaction size estimation.
|
|
739
|
+
const uncompressedPublicKeysTripleCorrectionFactor = 32 * 3;
|
|
740
|
+
return (
|
|
741
|
+
// This is not quite accurate - if there is a mix of inputs scripts where some used
|
|
742
|
+
// compressed keys and some used uncompressed keys, we would overestimate the size.
|
|
743
|
+
// Since we don't have mixed input sets, this should not be an issue in practice.
|
|
744
|
+
(unspents_1.VirtualSizes.txP2shInputSize +
|
|
745
|
+
(params.containsUncompressedPublicKeys ? uncompressedPublicKeysTripleCorrectionFactor : 0)) *
|
|
746
|
+
params.nP2shInputs +
|
|
706
747
|
unspents_1.VirtualSizes.txP2shP2wshInputSize * (params.nP2shP2wshInputs || 0) +
|
|
707
748
|
unspents_1.VirtualSizes.txP2pkhInputSizeUncompressedKey * (params.nP2pkhInputs || 0) +
|
|
708
749
|
unspents_1.VirtualSizes.txP2pkhOutputSize * params.nOutputs +
|
|
709
750
|
// if the tx contains at least one segwit input, the tx overhead is increased by 1
|
|
710
751
|
unspents_1.VirtualSizes.txOverheadSize +
|
|
711
|
-
(params.nP2shP2wshInputs > 0 ? 1 : 0);
|
|
712
|
-
return estimatedSize;
|
|
752
|
+
(params.nP2shP2wshInputs > 0 ? 1 : 0));
|
|
713
753
|
};
|
|
714
754
|
/**
|
|
715
755
|
* Calculate the fee and estimated size in bytes for a transaction.
|
|
@@ -756,19 +796,19 @@ exports.signTransaction = function (params) {
|
|
|
756
796
|
let keychain = params.keychain; // duplicate so as to not mutate below
|
|
757
797
|
const validate = params.validate === undefined ? true : params.validate;
|
|
758
798
|
let privKey;
|
|
759
|
-
if (!
|
|
799
|
+
if (!lodash_1.default.isString(params.transactionHex)) {
|
|
760
800
|
throw new Error('expecting the transaction hex as a string');
|
|
761
801
|
}
|
|
762
802
|
if (!Array.isArray(params.unspents)) {
|
|
763
803
|
throw new Error('expecting the unspents array');
|
|
764
804
|
}
|
|
765
|
-
if (!
|
|
805
|
+
if (!lodash_1.default.isBoolean(validate)) {
|
|
766
806
|
throw new Error('expecting validate to be a boolean');
|
|
767
807
|
}
|
|
768
|
-
let network = sdk_core_1.getNetwork();
|
|
769
|
-
const enableBCH =
|
|
770
|
-
if (!
|
|
771
|
-
if (
|
|
808
|
+
let network = (0, sdk_core_1.getNetwork)();
|
|
809
|
+
const enableBCH = lodash_1.default.isBoolean(params.forceBCH) && params.forceBCH === true;
|
|
810
|
+
if (!lodash_1.default.isObject(keychain) || !lodash_1.default.isString(keychain.xprv)) {
|
|
811
|
+
if (lodash_1.default.isString(params.signingKey)) {
|
|
772
812
|
privKey = utxolib.ECPair.fromWIF(params.signingKey, network);
|
|
773
813
|
keychain = undefined;
|
|
774
814
|
}
|
|
@@ -791,12 +831,12 @@ exports.signTransaction = function (params) {
|
|
|
791
831
|
throw new Error('length of unspents array should equal to the number of transaction inputs');
|
|
792
832
|
}
|
|
793
833
|
// decorate transaction with input values for TransactionBuilder instantiation
|
|
794
|
-
const isUtxoTx =
|
|
795
|
-
const areValidUnspents =
|
|
834
|
+
const isUtxoTx = lodash_1.default.isObject(transaction) && Array.isArray(transaction.ins);
|
|
835
|
+
const areValidUnspents = lodash_1.default.isObject(params) && Array.isArray(params.unspents);
|
|
796
836
|
if (isUtxoTx && areValidUnspents) {
|
|
797
837
|
// extend the transaction inputs with the values
|
|
798
|
-
const inputValues =
|
|
799
|
-
transaction.ins.map((currentItem, index) =>
|
|
838
|
+
const inputValues = lodash_1.default.map(params.unspents, (u) => lodash_1.default.pick(u, 'value'));
|
|
839
|
+
transaction.ins.map((currentItem, index) => lodash_1.default.extend(currentItem, inputValues[index]));
|
|
800
840
|
}
|
|
801
841
|
let rootExtKey;
|
|
802
842
|
if (keychain) {
|
|
@@ -822,7 +862,7 @@ exports.signTransaction = function (params) {
|
|
|
822
862
|
const chainPath = currentUnspent.chainPath;
|
|
823
863
|
if (rootExtKey) {
|
|
824
864
|
const { walletSubPath = '/0/0' } = keychain;
|
|
825
|
-
const path = sdk_core_1.sanitizeLegacyPath(keychain.path + walletSubPath + chainPath);
|
|
865
|
+
const path = (0, sdk_core_1.sanitizeLegacyPath)(keychain.path + walletSubPath + chainPath);
|
|
826
866
|
debug('derived user key path "%s" using keychain path "%s", walletSubPath "%s", keychain walletSubPath "%s" and chainPath "%s"', path, keychain.path, walletSubPath, keychain.walletSubPath, chainPath);
|
|
827
867
|
privKey = rootExtKey.derivePath(path);
|
|
828
868
|
}
|
|
@@ -839,12 +879,13 @@ exports.signTransaction = function (params) {
|
|
|
839
879
|
const witnessScript = currentUnspent.witnessScript ? Buffer.from(currentUnspent.witnessScript, 'hex') : undefined;
|
|
840
880
|
const sigHash = utxolib.bitgo.getDefaultSigHash(network);
|
|
841
881
|
txb.sign(index, privKey, subscript, sigHash, currentUnspent.value, witnessScript);
|
|
882
|
+
debug(`Signed transaction input ${index}`);
|
|
842
883
|
}
|
|
843
884
|
catch (e) {
|
|
844
885
|
// try fallback derivation path (see BG-46497)
|
|
845
886
|
let fallbackSigningSuccessful = false;
|
|
846
887
|
try {
|
|
847
|
-
const fallbackPath = sdk_core_1.sanitizeLegacyPath(keychain.path + chainPath);
|
|
888
|
+
const fallbackPath = (0, sdk_core_1.sanitizeLegacyPath)(keychain.path + chainPath);
|
|
848
889
|
debug('derived fallback user key path "%s" using keychain path "%s" and chainPath "%s"', fallbackPath, keychain.path, chainPath);
|
|
849
890
|
privKey = rootExtKey.derivePath(fallbackPath);
|
|
850
891
|
const witnessScript = currentUnspent.witnessScript
|
|
@@ -864,7 +905,7 @@ exports.signTransaction = function (params) {
|
|
|
864
905
|
};
|
|
865
906
|
e.message = `Failed to sign input #${index} - ${e.message} - ${JSON.stringify(e.result, null, 4)} - \n${e.stack}`;
|
|
866
907
|
debug('input sign failed: %s', e.message);
|
|
867
|
-
return
|
|
908
|
+
return bluebird_1.default.reject(e);
|
|
868
909
|
}
|
|
869
910
|
}
|
|
870
911
|
}
|
|
@@ -874,6 +915,7 @@ exports.signTransaction = function (params) {
|
|
|
874
915
|
const signatureCount = utxolib.bitgo
|
|
875
916
|
.getSignatureVerifications(partialTransaction, index, params.unspents[index].value)
|
|
876
917
|
.filter((v) => v.signedBy !== undefined).length;
|
|
918
|
+
debug(`Signature count for input ${index}: ${signatureCount}`);
|
|
877
919
|
if (signatureCount < 1) {
|
|
878
920
|
throw new Error('expected at least one valid signature');
|
|
879
921
|
}
|
|
@@ -882,8 +924,8 @@ exports.signTransaction = function (params) {
|
|
|
882
924
|
}
|
|
883
925
|
});
|
|
884
926
|
}
|
|
885
|
-
return
|
|
927
|
+
return bluebird_1.default.resolve({
|
|
886
928
|
transactionHex: partialTransaction.toHex(),
|
|
887
929
|
});
|
|
888
930
|
};
|
|
889
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"transactionBuilder.js","sourceRoot":"","sources":["../../../src/v1/transactionBuilder.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @hidden\n */\n\n/**\n */\n//\n// TransactionBuilder\n// A utility for building and signing transactions\n//\n// Copyright 2014, BitGo, Inc.  All Rights Reserved.\n//\n\nimport { bip32 } from '@bitgo-beta/utxo-lib';\nimport * as Bluebird from 'bluebird';\nimport * as utxolib from '@bitgo-beta/utxo-lib';\nimport * as _ from 'lodash';\nimport { VirtualSizes } from '@bitgo-beta/unspents';\nimport debugLib = require('debug');\nconst debug = debugLib('bitgo:v1:txb');\nimport { common, getAddressP2PKH, getNetwork, sanitizeLegacyPath } from '@bitgo-beta/sdk-core';\n\ninterface BaseOutput {\n  amount: number;\n  travelInfo?: any;\n}\n\ninterface AddressOutput extends BaseOutput {\n  address: string;\n}\n\ninterface ScriptOutput extends BaseOutput {\n  script: Buffer;\n}\n\ntype Output = AddressOutput | ScriptOutput;\n\ninterface BitGoUnspent {\n  value: number;\n  tx_hash: Buffer;\n  tx_output_n: number;\n}\n\n//\n// TransactionBuilder\n// @params:\n//   wallet:  a wallet object to send from\n//   recipients: array of recipient objects and the amount to send to each e.g. [{address: '38BKDNZbPcLogvVbcx2ekJ9E6Vv94DqDqw', amount: 1500}, {address: '36eL8yQqCn1HMRmVFFo49t2PJ3pai8wQam', amount: 2000}]\n//   fee: the fee to use with this transaction.  if not provided, a default, minimum fee will be used.\n//   feeRate: the amount of fee per kilobyte - optional - specify either fee, feeRate, or feeTxConfirmTarget but not more than one\n//   feeTxConfirmTarget: calculate the fees per kilobyte such that the transaction will be confirmed in this number of blocks\n//   maxFeeRate: The maximum fee per kb to use in satoshis, for safety purposes when using dynamic fees\n//   minConfirms: the minimum confirmations an output must have before spending\n//   forceChangeAtEnd: force the change address to be the last output\n//   changeAddress: specify the change address rather than generate a new one\n//   noSplitChange: set to true to disable automatic change splitting for purposes of unspent management\n//   targetWalletUnspents: specify a number of target unspents to maintain in the wallet (currently defaulted to 8 by the server)\n//   validate: extra verification of the change addresses, which is always done server-side and is redundant client-side (defaults true)\n//   minUnspentSize: The minimum size in satoshis of unspent to use (to prevent spending unspents worth less than fee added). Defaults to 0.\n//   feeSingleKeySourceAddress: Use this single key address to pay fees\n//   feeSingleKeyWIF: Use the address based on this private key to pay fees\n//   unspentsFetchParams: Extra parameters to use for fetching unspents for this transaction\nexports.createTransaction = function (params) {\n  const minConfirms = params.minConfirms || 0;\n  const validate = params.validate === undefined ? true : params.validate;\n  let recipients: { address: string; amount: number; script?: string; travelInfo?: any }[] = [];\n  let opReturns: { message: string; amount: number }[] = [];\n  let extraChangeAmounts: number[] = [];\n  let estTxSize: number;\n  let travelInfos;\n\n  // Sanity check the arguments passed in\n  if (\n    !_.isObject(params.wallet) ||\n    (params.fee && !_.isNumber(params.fee)) ||\n    (params.feeRate && !_.isNumber(params.feeRate)) ||\n    !_.isInteger(minConfirms) ||\n    (params.forceChangeAtEnd && !_.isBoolean(params.forceChangeAtEnd)) ||\n    (params.changeAddress && !_.isString(params.changeAddress)) ||\n    (params.noSplitChange && !_.isBoolean(params.noSplitChange)) ||\n    (params.targetWalletUnspents && !_.isInteger(params.targetWalletUnspents)) ||\n    (validate && !_.isBoolean(validate)) ||\n    (params.enforceMinConfirmsForChange && !_.isBoolean(params.enforceMinConfirmsForChange)) ||\n    (params.minUnspentSize && !_.isNumber(params.minUnspentSize)) ||\n    (params.maxFeeRate && !_.isNumber(params.maxFeeRate)) ||\n    // this should be an array and its length must be at least 1\n    (params.unspents && (!Array.isArray(params.unspents) || params.unspents.length < 1)) ||\n    (params.feeTxConfirmTarget && !_.isInteger(params.feeTxConfirmTarget)) ||\n    (params.instant && !_.isBoolean(params.instant)) ||\n    (params.bitgoFee && !_.isObject(params.bitgoFee)) ||\n    (params.unspentsFetchParams && !_.isObject(params.unspentsFetchParams))\n  ) {\n    throw new Error('invalid argument');\n  }\n\n  const bitgo = params.wallet.bitgo;\n  const constants = bitgo.getConstants();\n  const network = getNetwork(common.Environments[bitgo.getEnv()].network);\n\n  // The user can specify a seperate, single-key wallet for the purposes of paying miner's fees\n  // When creating a transaction this can be specified as an input address or the private key in WIF\n  let feeSingleKeySourceAddress;\n  let feeSingleKeyInputAmount = 0;\n  if (params.feeSingleKeySourceAddress) {\n    try {\n      utxolib.address.fromBase58Check(params.feeSingleKeySourceAddress, network);\n      feeSingleKeySourceAddress = params.feeSingleKeySourceAddress;\n    } catch (e) {\n      throw new Error('invalid bitcoin address: ' + params.feeSingleKeySourceAddress);\n    }\n  }\n\n  if (params.feeSingleKeyWIF) {\n    const feeSingleKey = utxolib.ECPair.fromWIF(params.feeSingleKeyWIF, network as utxolib.BitcoinJSNetwork);\n    feeSingleKeySourceAddress = getAddressP2PKH(feeSingleKey);\n    // If the user specifies both, check to make sure the feeSingleKeySourceAddress corresponds to the address of feeSingleKeyWIF\n    if (params.feeSingleKeySourceAddress && params.feeSingleKeySourceAddress !== feeSingleKeySourceAddress) {\n      throw new Error(\n        'feeSingleKeySourceAddress: ' +\n          params.feeSingleKeySourceAddress +\n          ' did not correspond to address of feeSingleKeyWIF: ' +\n          feeSingleKeySourceAddress\n      );\n    }\n  }\n\n  if (!_.isObject(params.recipients)) {\n    throw new Error('recipients must be array of { address: abc, amount: 100000 } objects');\n  }\n\n  let feeParamsDefined = 0;\n  if (!_.isUndefined(params.fee)) {\n    feeParamsDefined++;\n  }\n\n  if (!_.isUndefined(params.feeRate)) {\n    feeParamsDefined++;\n  }\n\n  if (!_.isUndefined(params.feeTxConfirmTarget)) {\n    feeParamsDefined++;\n  }\n\n  if (feeParamsDefined > 1) {\n    throw new Error('cannot specify more than one of fee, feeRate and feeTxConfirmTarget');\n  }\n\n  if (_.isUndefined(params.maxFeeRate)) {\n    params.maxFeeRate = constants.maxFeeRate;\n  }\n\n  // Convert the old format of params.recipients (dictionary of address:amount) to new format: { destinationAddress, amount }\n  if (!(params.recipients instanceof Array)) {\n    recipients = [];\n    Object.keys(params.recipients).forEach(function (destinationAddress) {\n      const amount = params.recipients[destinationAddress];\n      recipients.push({ address: destinationAddress, amount: amount });\n    });\n  } else {\n    recipients = params.recipients;\n  }\n\n  if (params.opReturns) {\n    if (!(params.opReturns instanceof Array)) {\n      opReturns = [];\n      Object.keys(params.opReturns).forEach(function (message) {\n        const amount = params.opReturns[message];\n        opReturns.push({ message, amount });\n      });\n    } else {\n      opReturns = params.opReturns;\n    }\n  }\n\n  if (recipients.length === 0 && opReturns.length === 0) {\n    throw new Error('must have at least one recipient');\n  }\n\n  let fee = params.fee;\n  let feeRate = params.feeRate;\n\n  // Flag indicating whether this class will compute the fee\n  const shouldComputeBestFee = _.isUndefined(fee);\n\n  let totalOutputAmount = 0;\n\n  recipients.forEach(function (recipient) {\n    if (_.isString(recipient.address)) {\n      try {\n        utxolib.address.fromBase58Check(recipient.address, network);\n      } catch (e) {\n        throw new Error('invalid bitcoin address: ' + recipient.address);\n      }\n      if (!!recipient.script) {\n        // A script was provided as well - validate that the address corresponds to that\n        if (utxolib.address.toOutputScript(recipient.address, network).toString('hex') !== recipient.script) {\n          throw new Error(\n            'both script and address provided but they did not match: ' + recipient.address + ' ' + recipient.script\n          );\n        }\n      }\n    }\n    if (!_.isInteger(recipient.amount) || recipient.amount < 0) {\n      throw new Error('invalid amount for ' + recipient.address + ': ' + recipient.amount);\n    }\n    totalOutputAmount += recipient.amount;\n  });\n\n  opReturns.forEach(function (opReturn) {\n    totalOutputAmount += opReturn.amount;\n  });\n\n  let bitgoFeeInfo = params.bitgoFee;\n  if (bitgoFeeInfo && (!_.isInteger(bitgoFeeInfo.amount) || !_.isString(bitgoFeeInfo.address))) {\n    throw new Error('invalid bitgoFeeInfo');\n  }\n\n  // The total amount needed for this transaction.\n  let totalAmount = totalOutputAmount + (fee || 0);\n\n  // The list of unspent transactions being used in this transaction.\n  let unspents;\n\n  // the total number of unspents on this wallet\n  let totalUnspentsCount;\n\n  // the number of unspents we fetched from the server, before filtering\n  let fetchedUnspentsCount;\n\n  // The list of unspent transactions being used with zero-confirmations\n  let zeroConfUnspentTxIds;\n\n  // The sum of the input values for this transaction.\n  let inputAmount;\n\n  let changeOutputs: Output[] = [];\n\n  // The transaction.\n  let transaction = utxolib.bitgo.createTransactionBuilderForNetwork(network);\n\n  const getBitGoFee = function () {\n    return Bluebird.try(function () {\n      if (bitgoFeeInfo) {\n        return;\n      }\n      return params.wallet.getBitGoFee({ amount: totalOutputAmount, instant: params.instant }).then(function (result) {\n        if (result && result.fee > 0) {\n          bitgoFeeInfo = {\n            amount: result.fee,\n          };\n        }\n      });\n    }).then(function () {\n      if (bitgoFeeInfo && bitgoFeeInfo.amount > 0) {\n        totalAmount += bitgoFeeInfo.amount;\n      }\n    });\n  };\n\n  const getBitGoFeeAddress = function () {\n    return Bluebird.try(function () {\n      // If we don't have bitgoFeeInfo, or address is already set, don't get a new one\n      if (!bitgoFeeInfo || bitgoFeeInfo.address) {\n        return;\n      }\n      return bitgo.getBitGoFeeAddress().then(function (result) {\n        bitgoFeeInfo.address = result.address;\n      });\n    });\n  };\n\n  // Get a dynamic fee estimate from the BitGo server if feeTxConfirmTarget\n  // is specified or if no fee-related params are specified\n  const getDynamicFeeRateEstimate = function () {\n    if (params.feeTxConfirmTarget || !feeParamsDefined) {\n      return bitgo\n        .estimateFee({\n          numBlocks: params.feeTxConfirmTarget,\n          maxFee: params.maxFeeRate,\n          inputs: zeroConfUnspentTxIds,\n          txSize: estTxSize,\n          cpfpAware: true,\n        })\n        .then(function (result) {\n          const estimatedFeeRate = result.cpfpFeePerKb;\n          const minimum = params.instant\n            ? Math.max(constants.minFeeRate, constants.minInstantFeeRate)\n            : constants.minFeeRate;\n          // 5 satoshis per byte\n          // it is worth noting that the padding only applies when the threshold is crossed, but not when the delta is less than the padding\n          const padding = 5000;\n          if (estimatedFeeRate < minimum) {\n            console.log(\n              new Date() +\n                ': Error when estimating fee for send from ' +\n                params.wallet.id() +\n                ', it was too low - ' +\n                estimatedFeeRate\n            );\n            feeRate = minimum + padding;\n          } else if (estimatedFeeRate > params.maxFeeRate) {\n            feeRate = params.maxFeeRate - padding;\n          } else {\n            feeRate = estimatedFeeRate;\n          }\n          return feeRate;\n        })\n        .catch(function (e) {\n          // sanity check failed on tx size\n          if (_.includes(e.message, 'invalid txSize')) {\n            return Bluebird.reject(e);\n          } else {\n            // couldn't estimate the fee, proceed using the default\n            feeRate = constants.fallbackFeeRate;\n            console.log('Error estimating fee for send from ' + params.wallet.id() + ': ' + e.message);\n            return Bluebird.resolve();\n          }\n        });\n    }\n  };\n\n  // Get the unspents for the sending wallet.\n  const getUnspents = function () {\n    if (params.unspents) {\n      // we just wanna use custom unspents\n      unspents = params.unspents;\n      return;\n    }\n\n    // Get enough unspents for the requested amount\n    const options = _.merge({}, params.unspentsFetchParams || {}, {\n      target: totalAmount,\n      minSize: params.minUnspentSize || 0,\n      instant: params.instant, // insist on instant unspents only\n      targetWalletUnspents: params.targetWalletUnspents,\n    });\n    if (params.instant) {\n      options.instant = params.instant; // insist on instant unspents only\n    }\n\n    return params.wallet.unspentsPaged(options).then(function (results) {\n      totalUnspentsCount = results.total;\n      fetchedUnspentsCount = results.count;\n      unspents = results.unspents.filter(function (u) {\n        const confirms = u.confirmations || 0;\n        if (!params.enforceMinConfirmsForChange && u.isChange) {\n          return true;\n        }\n        return confirms >= minConfirms;\n      });\n\n      // abort early if there's no viable unspents, because it won't be possible to create the txn later\n      if (unspents.length === 0) {\n        throw Error('0 unspents available for transaction creation');\n      }\n\n      // create array of unconfirmed unspent ID strings of the form \"txHash:outputIndex\"\n      zeroConfUnspentTxIds = _(results.unspents)\n        .filter(function (u) {\n          return !u.confirmations;\n        })\n        .map(function (u) {\n          return u.tx_hash + ':' + u.tx_output_n;\n        })\n        .value();\n      if (_.isEmpty(zeroConfUnspentTxIds)) {\n        // we don't want to pass an empty array of inputs to the server, because it assumes if the\n        // inputs arguments exists, it contains values\n        zeroConfUnspentTxIds = undefined;\n      }\n\n      // For backwards compatibility, respect the old splitChangeSize=0 parameter\n      if (!params.noSplitChange && params.splitChangeSize !== 0) {\n        extraChangeAmounts = results.extraChangeAmounts || [];\n      }\n    });\n  };\n\n  // Get the unspents for the single key fee address\n  let feeSingleKeyUnspents: BitGoUnspent[] = [];\n  const getUnspentsForSingleKey = function () {\n    if (feeSingleKeySourceAddress) {\n      let feeTarget = 0.01e8;\n      if (params.instant) {\n        feeTarget += totalAmount * 0.001;\n      }\n      return bitgo\n        .get(bitgo.url('/address/' + feeSingleKeySourceAddress + '/unspents?target=' + feeTarget))\n        .then(function (response) {\n          if (response.body.total <= 0) {\n            throw new Error('No unspents available in single key fee source');\n          }\n          feeSingleKeyUnspents = response.body.unspents;\n        });\n    }\n  };\n\n  let minerFeeInfo: any = {};\n  let txInfo: any = {};\n\n  // Iterate unspents, sum the inputs, and save _inputs with the total\n  // input amount and final list of inputs to use with the transaction.\n  let feeSingleKeyUnspentsUsed: BitGoUnspent[] = [];\n\n  const collectInputs = function () {\n    if (!unspents.length) {\n      throw new Error('no unspents available on wallet');\n    }\n    inputAmount = 0;\n\n    // Calculate the cost of spending a single input, i.e. the smallest economical unspent value\n    return Bluebird.try(function () {\n      if (_.isNumber(params.feeRate) || _.isNumber(params.originalFeeRate)) {\n        return !_.isUndefined(params.feeRate) ? params.feeRate : params.originalFeeRate;\n      } else {\n        return bitgo\n          .estimateFee({\n            numBlocks: params.feeTxConfirmTarget,\n            maxFee: params.maxFeeRate,\n          })\n          .then(function (feeRateEstimate) {\n            return feeRateEstimate.feePerKb;\n          });\n      }\n    })\n      .then(function (feeRate) {\n        // Don't spend inputs that cannot pay for their own cost.\n        let minInputValue = 0;\n        if (_.isInteger(params.minUnspentSize)) {\n          minInputValue = params.minUnspentSize;\n        }\n\n        let prunedUnspentCount = 0;\n        const originalUnspentCount = unspents.length;\n        unspents = _.filter(unspents, function (unspent) {\n          const isSegwitInput = !!unspent.witnessScript;\n          const currentInputSize = isSegwitInput ? VirtualSizes.txP2shP2wshInputSize : VirtualSizes.txP2shInputSize;\n          const feeBasedMinInputValue = (feeRate * currentInputSize) / 1000;\n          const currentMinInputValue = Math.max(minInputValue, feeBasedMinInputValue);\n          if (currentMinInputValue > unspent.value) {\n            // pruning unspent\n            const pruneDetails = {\n              generalMinInputValue: minInputValue,\n              feeBasedMinInputValue,\n              currentMinInputValue,\n              feeRate,\n              inputSize: currentInputSize,\n              unspent: unspent,\n            };\n            console.log(`pruning unspent: ${JSON.stringify(pruneDetails, null, 4)}`);\n            prunedUnspentCount++;\n            return false;\n          }\n          return true;\n        });\n\n        if (prunedUnspentCount > 0) {\n          console.log(`pruned ${prunedUnspentCount} out of ${originalUnspentCount} unspents`);\n        }\n\n        if (unspents.length === 0) {\n          throw new Error('insufficient funds');\n        }\n        let segwitInputCount = 0;\n        unspents.every(function (unspent) {\n          if (unspent.witnessScript) {\n            segwitInputCount++;\n          }\n          inputAmount += unspent.value;\n          transaction.addInput(unspent.tx_hash, unspent.tx_output_n, 0xffffffff);\n\n          return inputAmount < (feeSingleKeySourceAddress ? totalOutputAmount : totalAmount);\n        });\n\n        // if paying fees from an external single key wallet, add the inputs\n        if (feeSingleKeySourceAddress) {\n          // collect the amount used in the fee inputs so we can get change later\n          feeSingleKeyInputAmount = 0;\n          feeSingleKeyUnspentsUsed = [];\n          feeSingleKeyUnspents.every(function (unspent) {\n            feeSingleKeyInputAmount += unspent.value;\n            inputAmount += unspent.value;\n            transaction.addInput(unspent.tx_hash, unspent.tx_output_n);\n            feeSingleKeyUnspentsUsed.push(unspent);\n            // use the fee wallet to pay miner fees and potentially instant fees\n            return feeSingleKeyInputAmount < fee + (bitgoFeeInfo ? bitgoFeeInfo.amount : 0);\n          });\n        }\n\n        txInfo = {\n          nP2shInputs: transaction.tx.ins.length - (feeSingleKeySourceAddress ? 1 : 0) - segwitInputCount,\n          nP2shP2wshInputs: segwitInputCount,\n          nP2pkhInputs: feeSingleKeySourceAddress ? 1 : 0,\n          // add single key source address change\n          nOutputs:\n            recipients.length +\n            1 + // recipients and change\n            extraChangeAmounts.length + // extra change splitting\n            (bitgoFeeInfo && bitgoFeeInfo.amount > 0 ? 1 : 0) + // add output for bitgo fee\n            (feeSingleKeySourceAddress ? 1 : 0),\n        };\n\n        estTxSize = estimateTransactionSize({\n          nP2shInputs: txInfo.nP2shInputs,\n          nP2shP2wshInputs: txInfo.nP2shP2wshInputs,\n          nP2pkhInputs: txInfo.nP2pkhInputs,\n          nOutputs: txInfo.nOutputs,\n        });\n      })\n      .then(getDynamicFeeRateEstimate)\n      .then(function () {\n        minerFeeInfo = exports.calculateMinerFeeInfo({\n          bitgo: params.wallet.bitgo,\n          feeRate: feeRate,\n          nP2shInputs: txInfo.nP2shInputs,\n          nP2shP2wshInputs: txInfo.nP2shP2wshInputs,\n          nP2pkhInputs: txInfo.nP2pkhInputs,\n          nOutputs: txInfo.nOutputs,\n        });\n\n        if (shouldComputeBestFee) {\n          const approximateFee = minerFeeInfo.fee;\n          const shouldRecurse = _.isUndefined(fee) || approximateFee > fee;\n          fee = approximateFee;\n          // Recompute totalAmount from scratch\n          totalAmount = fee + totalOutputAmount;\n          if (bitgoFeeInfo) {\n            totalAmount += bitgoFeeInfo.amount;\n          }\n          if (shouldRecurse) {\n            // if fee changed, re-collect inputs\n            inputAmount = 0;\n            transaction = utxolib.bitgo.createTransactionBuilderForNetwork(network);\n            return collectInputs();\n          }\n        }\n\n        const totalFee = fee + (bitgoFeeInfo ? bitgoFeeInfo.amount : 0);\n\n        if (feeSingleKeySourceAddress) {\n          const summedSingleKeyUnspents = _.sumBy(feeSingleKeyUnspents, 'value');\n          if (totalFee > summedSingleKeyUnspents) {\n            const err: any = new Error(\n              'Insufficient fee amount available in single key fee source: ' + summedSingleKeyUnspents\n            );\n            err.result = {\n              fee: fee,\n              feeRate: feeRate,\n              estimatedSize: minerFeeInfo.size,\n              available: inputAmount,\n              bitgoFee: bitgoFeeInfo,\n              txInfo: txInfo,\n            };\n            return Bluebird.reject(err);\n          }\n        }\n\n        if (inputAmount < (feeSingleKeySourceAddress ? totalOutputAmount : totalAmount)) {\n          // The unspents we're using for inputs do not have sufficient value on them to\n          // satisfy the user's requested spend amount. That may be because the wallet's balance\n          // is simply too low, or it might be that the wallet's balance is sufficient but\n          // we didn't fetch enough unspents. Too few unspents could result from the wallet\n          // having many small unspents and we hit our limit on the number of inputs we can use\n          // in a txn, or it might have been that the filters the user passed in (like minConfirms)\n          // disqualified too many of the unspents\n          let err;\n          if (totalUnspentsCount === fetchedUnspentsCount) {\n            // we fetched every unspent the wallet had, but it still wasn't enough\n            err = new Error('Insufficient funds');\n          } else {\n            // we weren't able to fetch all the unspents on the wallet\n            err = new Error(\n              `Transaction size too large due to too many unspents. Can send only ${inputAmount} satoshis in this transaction`\n            );\n          }\n          err.result = {\n            fee: fee,\n            feeRate: feeRate,\n            estimatedSize: minerFeeInfo.size,\n            available: inputAmount,\n            bitgoFee: bitgoFeeInfo,\n            txInfo: txInfo,\n          };\n          return Bluebird.reject(err);\n        }\n      });\n  };\n\n  // Add the outputs for this transaction.\n  const collectOutputs = function () {\n    if (minerFeeInfo.size >= 90000) {\n      throw new Error('transaction too large: estimated size ' + minerFeeInfo.size + ' bytes');\n    }\n\n    const outputs: Output[] = [];\n\n    recipients.forEach(function (recipient) {\n      let script;\n      if (_.isString(recipient.address)) {\n        script = utxolib.address.toOutputScript(recipient.address, network);\n      } else if (_.isObject(recipient.script)) {\n        script = recipient.script;\n      } else {\n        throw new Error('neither recipient address nor script was provided');\n      }\n\n      // validate travelInfo if it exists\n      let travelInfo;\n      if (!_.isEmpty(recipient.travelInfo)) {\n        travelInfo = recipient.travelInfo;\n        // Better to avoid trouble now, before tx is created\n        bitgo.travelRule().validateTravelInfo(travelInfo);\n      }\n\n      outputs.push({\n        script: script,\n        amount: recipient.amount,\n        travelInfo: travelInfo,\n      });\n    });\n\n    opReturns.forEach(function ({ message, amount }) {\n      const script = utxolib.script.fromASM('OP_RETURN ' + Buffer.from(message).toString('hex'));\n      outputs.push({ script, amount });\n    });\n\n    const getChangeOutputs = function (changeAmount: number): Output[] | Bluebird<Output[]> {\n      if (changeAmount < 0) {\n        throw new Error('negative change amount: ' + changeAmount);\n      }\n\n      const result: Output[] = [];\n      // if we paid fees from a single key wallet, return the fee change first\n      if (feeSingleKeySourceAddress) {\n        const feeSingleKeyWalletChangeAmount =\n          feeSingleKeyInputAmount - (fee + (bitgoFeeInfo ? bitgoFeeInfo.amount : 0));\n        if (feeSingleKeyWalletChangeAmount >= constants.minOutputSize) {\n          result.push({ address: feeSingleKeySourceAddress, amount: feeSingleKeyWalletChangeAmount });\n          changeAmount = changeAmount - feeSingleKeyWalletChangeAmount;\n        }\n      }\n\n      if (changeAmount < constants.minOutputSize) {\n        // Give it to the miners\n        return result;\n      }\n\n      if (params.wallet.type() === 'safe') {\n        return params.wallet.addresses().then(function (response) {\n          result.push({ address: response.addresses[0].address, amount: changeAmount });\n          return result;\n        });\n      }\n\n      let extraChangeTotal = _.sum(extraChangeAmounts);\n      // Sanity check\n      if (extraChangeTotal > changeAmount) {\n        extraChangeAmounts = [];\n        extraChangeTotal = 0;\n      }\n\n      // copy and add remaining change amount\n      const allChangeAmounts = extraChangeAmounts.slice(0);\n      allChangeAmounts.push(changeAmount - extraChangeTotal);\n\n      // Recursive async func to add all change outputs\n      const addChangeOutputs = function (): Output[] | Bluebird<Output[]> {\n        const thisAmount = allChangeAmounts.shift();\n        if (!thisAmount) {\n          return result;\n        }\n        return Bluebird.try(function () {\n          if (params.changeAddress) {\n            // If user passed a change address, use it for all outputs\n            return params.changeAddress;\n          } else {\n            // Otherwise create a new address per output, for privacy\n            // determine if segwit or not\n            const changeChain = params.wallet.getChangeChain(params);\n            return params.wallet.createAddress({ chain: changeChain, validate: validate }).then(function (result) {\n              return result.address;\n            });\n          }\n        }).then(function (address) {\n          result.push({ address: address, amount: thisAmount });\n          return addChangeOutputs();\n        });\n      };\n\n      return addChangeOutputs();\n    };\n\n    // Add change output(s) and instant fee output if applicable\n    return Bluebird.try(function () {\n      return getChangeOutputs(inputAmount - totalAmount);\n    }).then(function (result) {\n      changeOutputs = result;\n      const extraOutputs = changeOutputs.concat([]); // copy the array\n      if (bitgoFeeInfo && bitgoFeeInfo.amount > 0) {\n        extraOutputs.push(bitgoFeeInfo);\n      }\n      extraOutputs.forEach(function (output) {\n        if ((output as AddressOutput).address) {\n          (output as ScriptOutput).script = utxolib.address.toOutputScript((output as AddressOutput).address, network);\n        }\n\n        // decide where to put the outputs - default is to randomize unless forced to end\n        const outputIndex = params.forceChangeAtEnd ? outputs.length : _.random(0, outputs.length);\n        outputs.splice(outputIndex, 0, output);\n      });\n\n      // Add all outputs to the transaction\n      outputs.forEach(function (output) {\n        transaction.addOutput((output as ScriptOutput).script, output.amount);\n      });\n\n      travelInfos = _(outputs)\n        .map(function (output, index) {\n          const result = output.travelInfo;\n          if (!result) {\n            return undefined;\n          }\n          result.outputIndex = index;\n          return result;\n        })\n        .filter()\n        .value();\n    });\n  };\n\n  // Serialize the transaction, returning what is needed to sign it\n  const serialize = function () {\n    // only need to return the unspents that were used and just the chainPath, redeemScript, and instant flag\n    const pickedUnspents: any = _.map(unspents, function (unspent) {\n      return _.pick(unspent, ['chainPath', 'redeemScript', 'instant', 'witnessScript', 'script', 'value']);\n    });\n    const prunedUnspents = _.slice(pickedUnspents, 0, transaction.tx.ins.length - feeSingleKeyUnspentsUsed.length);\n    _.each(feeSingleKeyUnspentsUsed, function (feeUnspent) {\n      prunedUnspents.push({ redeemScript: false, chainPath: false }); // mark as false to signify a non-multisig address\n    });\n    const result: any = {\n      transactionHex: transaction.buildIncomplete().toHex(),\n      unspents: prunedUnspents,\n      fee: fee,\n      changeAddresses: changeOutputs.map(function (co) {\n        return _.pick(co, ['address', 'path', 'amount']);\n      }),\n      walletId: params.wallet.id(),\n      walletKeychains: params.wallet.keychains,\n      feeRate: feeRate,\n      instant: params.instant,\n      bitgoFee: bitgoFeeInfo,\n      estimatedSize: minerFeeInfo.size,\n      txInfo: txInfo,\n      travelInfos: travelInfos,\n    };\n\n    // Add for backwards compatibility\n    if (result.instant && bitgoFeeInfo) {\n      result.instantFee = _.pick(bitgoFeeInfo, ['amount', 'address']);\n    }\n\n    return result;\n  };\n\n  return Bluebird.try(function () {\n    return getBitGoFee();\n  })\n    .then(function () {\n      return Bluebird.all([getBitGoFeeAddress(), getUnspents(), getUnspentsForSingleKey()]);\n    })\n    .then(collectInputs)\n    .then(collectOutputs)\n    .then(serialize);\n};\n\n/**\n * Estimate the size of a transaction in bytes based on the number of\n * inputs and outputs present.\n * @params params {\n *   nP2shInputs: number of P2SH (multisig) inputs\n *   nP2pkhInputs: number of P2PKH (single sig) inputs\n *   nOutputs: number of outputs\n * }\n *\n * @returns size: estimated size of the transaction in bytes\n */\nconst estimateTransactionSize = function (params) {\n  if (!_.isInteger(params.nP2shInputs) || params.nP2shInputs < 0) {\n    throw new Error('expecting positive nP2shInputs');\n  }\n  if (!_.isInteger(params.nP2pkhInputs) || params.nP2pkhInputs < 0) {\n    throw new Error('expecting positive nP2pkhInputs to be numeric');\n  }\n  if (!_.isInteger(params.nP2shP2wshInputs) || params.nP2shP2wshInputs < 0) {\n    throw new Error('expecting positive nP2shP2wshInputs to be numeric');\n  }\n  if (params.nP2shInputs + params.nP2shP2wshInputs < 1) {\n    throw new Error('expecting at least one nP2shInputs or nP2shP2wshInputs');\n  }\n  if (!_.isInteger(params.nOutputs) || params.nOutputs < 1) {\n    throw new Error('expecting positive nOutputs');\n  }\n\n  const estimatedSize =\n    VirtualSizes.txP2shInputSize * params.nP2shInputs +\n    VirtualSizes.txP2shP2wshInputSize * (params.nP2shP2wshInputs || 0) +\n    VirtualSizes.txP2pkhInputSizeUncompressedKey * (params.nP2pkhInputs || 0) +\n    VirtualSizes.txP2pkhOutputSize * params.nOutputs +\n    // if the tx contains at least one segwit input, the tx overhead is increased by 1\n    VirtualSizes.txOverheadSize +\n    (params.nP2shP2wshInputs > 0 ? 1 : 0);\n\n  return estimatedSize;\n};\n\n/**\n * Calculate the fee and estimated size in bytes for a transaction.\n * @params params {\n *   bitgo: bitgo object\n *   feeRate: satoshis per kilobyte\n *   nP2shInputs: number of P2SH (multisig) inputs\n *   nP2pkhInputs: number of P2PKH (single sig) inputs\n *   nOutputs: number of outputs\n * }\n *\n * @returns {\n *   size: estimated size of the transaction in bytes\n *   fee: estimated fee in satoshis for the transaction\n *   feeRate: fee rate that was used to estimate the fee for the transaction\n * }\n */\nexports.calculateMinerFeeInfo = function (params) {\n  const feeRateToUse = params.feeRate || params.bitgo.getConstants().fallbackFeeRate;\n  const estimatedSize = estimateTransactionSize(params);\n\n  return {\n    size: estimatedSize,\n    fee: Math.ceil((estimatedSize * feeRateToUse) / 1000),\n    feeRate: feeRateToUse,\n  };\n};\n\n/*\n * Given a transaction hex, unspent information (chain path and redeem scripts), and the keychain xprv,\n * perform key derivation and sign the inputs in the transaction based on the unspent information provided\n *\n * @params:\n *  transactionHex serialized form of the transaction in hex\n *  unspents array of unspent information, where each unspent is a chainPath and redeemScript with the same\n *  index as the inputs in the transactionHex\n *  keychain Keychain containing the xprv to sign with. For legacy support of safe wallets, keychain can\n also be a WIF private key.\n *  signingKey private key in WIF for safe wallets, when keychain is unavailable\n *  validate client-side signature verification - can be disabled for improved performance (signatures\n *           are still validated server-side).\n *  feeSingleKeyWIF Use the address based on this private key to pay fees\n * @returns {*}\n */\nexports.signTransaction = function (params) {\n  let keychain = params.keychain; // duplicate so as to not mutate below\n\n  const validate = params.validate === undefined ? true : params.validate;\n  let privKey;\n  if (!_.isString(params.transactionHex)) {\n    throw new Error('expecting the transaction hex as a string');\n  }\n  if (!Array.isArray(params.unspents)) {\n    throw new Error('expecting the unspents array');\n  }\n  if (!_.isBoolean(validate)) {\n    throw new Error('expecting validate to be a boolean');\n  }\n  let network = getNetwork();\n  const enableBCH = _.isBoolean(params.forceBCH) && params.forceBCH === true;\n\n  if (!_.isObject(keychain) || !_.isString((keychain as any).xprv)) {\n    if (_.isString(params.signingKey)) {\n      privKey = utxolib.ECPair.fromWIF(params.signingKey, network as utxolib.BitcoinJSNetwork);\n      keychain = undefined;\n    } else {\n      throw new Error('expecting the keychain object with xprv');\n    }\n  }\n\n  let feeSingleKey;\n  if (params.feeSingleKeyWIF) {\n    feeSingleKey = utxolib.ECPair.fromWIF(params.feeSingleKeyWIF, network as utxolib.BitcoinJSNetwork);\n  }\n\n  debug('Network: %O', network);\n\n  if (enableBCH) {\n    debug('Enabling BCH…');\n    network = utxolib.networks.bitcoincash;\n    debug('New network: %O', network);\n  }\n\n  const transaction = utxolib.bitgo.createTransactionFromHex(params.transactionHex, network);\n  if (transaction.ins.length !== params.unspents.length) {\n    throw new Error('length of unspents array should equal to the number of transaction inputs');\n  }\n\n  // decorate transaction with input values for TransactionBuilder instantiation\n  const isUtxoTx = _.isObject(transaction) && Array.isArray((transaction as any).ins);\n  const areValidUnspents = _.isObject(params) && Array.isArray((params as any).unspents);\n  if (isUtxoTx && areValidUnspents) {\n    // extend the transaction inputs with the values\n    const inputValues = _.map((params as any).unspents, (u) => _.pick(u, 'value'));\n    transaction.ins.map((currentItem, index) => _.extend(currentItem, inputValues[index]));\n  }\n\n  let rootExtKey;\n  if (keychain) {\n    rootExtKey = bip32.fromBase58(keychain.xprv);\n  }\n\n  const txb = utxolib.bitgo.createTransactionBuilderFromTransaction(transaction);\n\n  for (let index = 0; index < txb.tx.ins.length; ++index) {\n    const currentUnspent = params.unspents[index];\n    if (currentUnspent.redeemScript === false) {\n      // this is the input from a single key fee address\n      if (!feeSingleKey) {\n        throw new Error('single key address used in input but feeSingleKeyWIF not provided');\n      }\n\n      if (enableBCH) {\n        feeSingleKey.network = network;\n      }\n\n      txb.sign(index, feeSingleKey);\n      continue;\n    }\n\n    if (currentUnspent.witnessScript && enableBCH) {\n      throw new Error('BCH does not support segwit inputs');\n    }\n\n    const chainPath = currentUnspent.chainPath;\n    if (rootExtKey) {\n      const { walletSubPath = '/0/0' } = keychain;\n      const path = sanitizeLegacyPath(keychain.path + walletSubPath + chainPath);\n      debug(\n        'derived user key path \"%s\" using keychain path \"%s\", walletSubPath \"%s\", keychain walletSubPath \"%s\" and chainPath \"%s\"',\n        path,\n        keychain.path,\n        walletSubPath,\n        keychain.walletSubPath,\n        chainPath\n      );\n      privKey = rootExtKey.derivePath(path);\n    }\n\n    privKey.network = network;\n\n    // subscript is the part of the output script after the OP_CODESEPARATOR.\n    // Since we are only ever signing p2sh outputs, which do not have\n    // OP_CODESEPARATORS, it is always the output script.\n    const subscript = Buffer.from(currentUnspent.redeemScript, 'hex');\n    currentUnspent.validationScript = subscript;\n\n    // In order to sign with bitcoinjs-lib, we must use its transaction\n    // builder, confusingly named the same exact thing as our transaction\n    // builder, but with inequivalent behavior.\n    try {\n      const witnessScript = currentUnspent.witnessScript ? Buffer.from(currentUnspent.witnessScript, 'hex') : undefined;\n      const sigHash = utxolib.bitgo.getDefaultSigHash(network);\n      txb.sign(index, privKey, subscript, sigHash, currentUnspent.value, witnessScript);\n    } catch (e) {\n      // try fallback derivation path (see BG-46497)\n      let fallbackSigningSuccessful = false;\n      try {\n        const fallbackPath = sanitizeLegacyPath(keychain.path + chainPath);\n        debug(\n          'derived fallback user key path \"%s\" using keychain path \"%s\" and chainPath \"%s\"',\n          fallbackPath,\n          keychain.path,\n          chainPath\n        );\n        privKey = rootExtKey.derivePath(fallbackPath);\n        const witnessScript = currentUnspent.witnessScript\n          ? Buffer.from(currentUnspent.witnessScript, 'hex')\n          : undefined;\n        const sigHash = utxolib.bitgo.getDefaultSigHash(network);\n        txb.sign(index, privKey, subscript, sigHash, currentUnspent.value, witnessScript);\n        fallbackSigningSuccessful = true;\n      } catch (fallbackError) {\n        debug('input sign failed for fallback path: %s', fallbackError.message);\n      }\n      // we need to know what's causing this\n      if (!fallbackSigningSuccessful) {\n        e.result = {\n          unspent: currentUnspent,\n        };\n        e.message = `Failed to sign input #${index} - ${e.message} - ${JSON.stringify(e.result, null, 4)} - \\n${\n          e.stack\n        }`;\n        debug('input sign failed: %s', e.message);\n        return Bluebird.reject(e);\n      }\n    }\n  }\n\n  const partialTransaction = txb.buildIncomplete();\n\n  if (validate) {\n    partialTransaction.ins.forEach((input, index) => {\n      const signatureCount = utxolib.bitgo\n        .getSignatureVerifications(partialTransaction, index, params.unspents[index].value)\n        .filter((v) => v.signedBy !== undefined).length;\n      if (signatureCount < 1) {\n        throw new Error('expected at least one valid signature');\n      }\n      if (params.fullLocalSigning && signatureCount < 2) {\n        throw new Error('fullLocalSigning set: expected at least two valid signatures');\n      }\n    });\n  }\n\n  return Bluebird.resolve({\n    transactionHex: partialTransaction.toHex(),\n  });\n};\n"]}
|
|
931
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"transactionBuilder.js","sourceRoot":"","sources":["../../../src/v1/transactionBuilder.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @hidden\n */\n\n/**\n */\n//\n// TransactionBuilder\n// A utility for building and signing transactions\n//\n// Copyright 2014, BitGo, Inc.  All Rights Reserved.\n//\n\nimport { bip32 } from '@bitgo-beta/utxo-lib';\nimport Bluebird from 'bluebird';\nimport * as utxolib from '@bitgo-beta/utxo-lib';\nimport _ from 'lodash';\nimport { VirtualSizes } from '@bitgo-beta/unspents';\nimport debugLib = require('debug');\nconst debug = debugLib('bitgo:v1:txb');\nimport { common, getAddressP2PKH, getNetwork, sanitizeLegacyPath } from '@bitgo-beta/sdk-core';\nimport { verifyAddress } from './verifyAddress';\n\ninterface BaseOutput {\n  amount: number;\n  travelInfo?: any;\n}\n\ninterface AddressOutput extends BaseOutput {\n  address: string;\n}\n\ninterface ScriptOutput extends BaseOutput {\n  script: Buffer;\n}\n\ntype Output = AddressOutput | ScriptOutput;\n\ninterface BitGoUnspent {\n  value: number;\n  tx_hash: Buffer;\n  tx_output_n: number;\n}\n\n//\n// TransactionBuilder\n// @params:\n//   wallet:  a wallet object to send from\n//   recipients: array of recipient objects and the amount to send to each e.g. [{address: '38BKDNZbPcLogvVbcx2ekJ9E6Vv94DqDqw', amount: 1500}, {address: '36eL8yQqCn1HMRmVFFo49t2PJ3pai8wQam', amount: 2000}]\n//   fee: the fee to use with this transaction.  if not provided, a default, minimum fee will be used.\n//   feeRate: the amount of fee per kilobyte - optional - specify either fee, feeRate, or feeTxConfirmTarget but not more than one\n//   feeTxConfirmTarget: calculate the fees per kilobyte such that the transaction will be confirmed in this number of blocks\n//   maxFeeRate: The maximum fee per kb to use in satoshis, for safety purposes when using dynamic fees\n//   minConfirms: the minimum confirmations an output must have before spending\n//   forceChangeAtEnd: force the change address to be the last output\n//   changeAddress: specify the change address rather than generate a new one\n//   noSplitChange: set to true to disable automatic change splitting for purposes of unspent management\n//   targetWalletUnspents: specify a number of target unspents to maintain in the wallet (currently defaulted to 8 by the server)\n//   validate: extra verification of the change addresses, which is always done server-side and is redundant client-side (defaults true)\n//   minUnspentSize: The minimum size in satoshis of unspent to use (to prevent spending unspents worth less than fee added). Defaults to 0.\n//   feeSingleKeySourceAddress: Use this single key address to pay fees\n//   feeSingleKeyWIF: Use the address based on this private key to pay fees\n//   unspentsFetchParams: Extra parameters to use for fetching unspents for this transaction\n//   unspents: array of unspent objects to use while constructing the transaction instead of fetching from the API\nexports.createTransaction = function (params) {\n  const minConfirms = params.minConfirms || 0;\n  const validate = params.validate === undefined ? true : params.validate;\n  let recipients: { address: string; amount: number; script?: string; travelInfo?: any }[] = [];\n  let opReturns: { message: string; amount: number }[] = [];\n  let extraChangeAmounts: number[] = [];\n  let estTxSize: number;\n  let travelInfos;\n\n  // Sanity check the arguments passed in\n  if (\n    !_.isObject(params.wallet) ||\n    (params.fee && !_.isNumber(params.fee)) ||\n    (params.feeRate && !_.isNumber(params.feeRate)) ||\n    !_.isInteger(minConfirms) ||\n    (params.forceChangeAtEnd && !_.isBoolean(params.forceChangeAtEnd)) ||\n    (params.changeAddress && !_.isString(params.changeAddress)) ||\n    (params.noSplitChange && !_.isBoolean(params.noSplitChange)) ||\n    (params.targetWalletUnspents && !_.isInteger(params.targetWalletUnspents)) ||\n    (validate && !_.isBoolean(validate)) ||\n    (params.enforceMinConfirmsForChange && !_.isBoolean(params.enforceMinConfirmsForChange)) ||\n    (params.minUnspentSize && !_.isNumber(params.minUnspentSize)) ||\n    (params.maxFeeRate && !_.isNumber(params.maxFeeRate)) ||\n    // this should be an array and its length must be at least 1\n    (params.unspents && (!Array.isArray(params.unspents) || params.unspents.length < 1)) ||\n    (params.feeTxConfirmTarget && !_.isInteger(params.feeTxConfirmTarget)) ||\n    (params.instant && !_.isBoolean(params.instant)) ||\n    (params.bitgoFee && !_.isObject(params.bitgoFee)) ||\n    (params.unspentsFetchParams && !_.isObject(params.unspentsFetchParams))\n  ) {\n    throw new Error('invalid argument');\n  }\n\n  const bitgo = params.wallet.bitgo;\n  const constants = bitgo.getConstants();\n  const network = getNetwork(common.Environments[bitgo.getEnv()].network);\n\n  // The user can specify a seperate, single-key wallet for the purposes of paying miner's fees\n  // When creating a transaction this can be specified as an input address or the private key in WIF\n  let feeSingleKeySourceAddress;\n  let feeSingleKeyInputAmount = 0;\n  if (params.feeSingleKeySourceAddress) {\n    try {\n      utxolib.address.fromBase58Check(params.feeSingleKeySourceAddress, network);\n      feeSingleKeySourceAddress = params.feeSingleKeySourceAddress;\n    } catch (e) {\n      throw new Error('invalid bitcoin address: ' + params.feeSingleKeySourceAddress);\n    }\n  }\n\n  if (params.feeSingleKeyWIF) {\n    const feeSingleKey = utxolib.ECPair.fromWIF(params.feeSingleKeyWIF, network as utxolib.BitcoinJSNetwork);\n    feeSingleKeySourceAddress = getAddressP2PKH(feeSingleKey);\n    // If the user specifies both, check to make sure the feeSingleKeySourceAddress corresponds to the address of feeSingleKeyWIF\n    if (params.feeSingleKeySourceAddress && params.feeSingleKeySourceAddress !== feeSingleKeySourceAddress) {\n      throw new Error(\n        'feeSingleKeySourceAddress: ' +\n          params.feeSingleKeySourceAddress +\n          ' did not correspond to address of feeSingleKeyWIF: ' +\n          feeSingleKeySourceAddress\n      );\n    }\n  }\n\n  if (!_.isObject(params.recipients)) {\n    throw new Error('recipients must be array of { address: abc, amount: 100000 } objects');\n  }\n\n  let feeParamsDefined = 0;\n  if (!_.isUndefined(params.fee)) {\n    feeParamsDefined++;\n  }\n\n  if (!_.isUndefined(params.feeRate)) {\n    feeParamsDefined++;\n  }\n\n  if (!_.isUndefined(params.feeTxConfirmTarget)) {\n    feeParamsDefined++;\n  }\n\n  if (feeParamsDefined > 1) {\n    throw new Error('cannot specify more than one of fee, feeRate and feeTxConfirmTarget');\n  }\n\n  if (_.isUndefined(params.maxFeeRate)) {\n    params.maxFeeRate = constants.maxFeeRate;\n  }\n\n  // Convert the old format of params.recipients (dictionary of address:amount) to new format: { destinationAddress, amount }\n  if (!(params.recipients instanceof Array)) {\n    recipients = [];\n    Object.keys(params.recipients).forEach(function (destinationAddress) {\n      const amount = params.recipients[destinationAddress];\n      recipients.push({ address: destinationAddress, amount: amount });\n    });\n  } else {\n    recipients = params.recipients;\n  }\n\n  if (params.opReturns) {\n    if (!(params.opReturns instanceof Array)) {\n      opReturns = [];\n      Object.keys(params.opReturns).forEach(function (message) {\n        const amount = params.opReturns[message];\n        opReturns.push({ message, amount });\n      });\n    } else {\n      opReturns = params.opReturns;\n    }\n  }\n\n  if (recipients.length === 0 && opReturns.length === 0) {\n    throw new Error('must have at least one recipient');\n  }\n\n  let fee = params.fee;\n  let feeRate = params.feeRate;\n\n  // Flag indicating whether this class will compute the fee\n  const shouldComputeBestFee = _.isUndefined(fee);\n\n  let totalOutputAmount = 0;\n\n  recipients.forEach(function (recipient) {\n    if (_.isString(recipient.address)) {\n      if (!verifyAddress(recipient.address, network)) {\n        throw new Error('invalid bitcoin address: ' + recipient.address);\n      }\n      if (!!recipient.script) {\n        // A script was provided as well - validate that the address corresponds to that\n        if (utxolib.address.toOutputScript(recipient.address, network).toString('hex') !== recipient.script) {\n          throw new Error(\n            'both script and address provided but they did not match: ' + recipient.address + ' ' + recipient.script\n          );\n        }\n      }\n    }\n    if (!_.isInteger(recipient.amount) || recipient.amount < 0) {\n      throw new Error('invalid amount for ' + recipient.address + ': ' + recipient.amount);\n    }\n    totalOutputAmount += recipient.amount;\n  });\n\n  opReturns.forEach(function (opReturn) {\n    totalOutputAmount += opReturn.amount;\n  });\n\n  let bitgoFeeInfo = params.bitgoFee;\n  if (bitgoFeeInfo && (!_.isInteger(bitgoFeeInfo.amount) || !_.isString(bitgoFeeInfo.address))) {\n    throw new Error('invalid bitgoFeeInfo');\n  }\n\n  // The total amount needed for this transaction.\n  let totalAmount = totalOutputAmount + (fee || 0);\n\n  // The list of unspent transactions being used in this transaction.\n  let unspents;\n\n  // the total number of unspents on this wallet\n  let totalUnspentsCount;\n\n  // the number of unspents we fetched from the server, before filtering\n  let fetchedUnspentsCount;\n\n  // The list of unspent transactions being used with zero-confirmations\n  let zeroConfUnspentTxIds;\n\n  // The sum of the input values for this transaction.\n  let inputAmount;\n\n  let changeOutputs: Output[] = [];\n\n  let containsUncompressedPublicKeys = false;\n\n  // The transaction.\n  let transaction = utxolib.bitgo.createTransactionBuilderForNetwork(network);\n\n  const getBitGoFee = function () {\n    return Bluebird.try(function () {\n      if (bitgoFeeInfo) {\n        return;\n      }\n      return params.wallet.getBitGoFee({ amount: totalOutputAmount, instant: params.instant }).then(function (result) {\n        if (result && result.fee > 0) {\n          bitgoFeeInfo = {\n            amount: result.fee,\n          };\n        }\n      });\n    }).then(function () {\n      if (bitgoFeeInfo && bitgoFeeInfo.amount > 0) {\n        totalAmount += bitgoFeeInfo.amount;\n      }\n    });\n  };\n\n  const getBitGoFeeAddress = function () {\n    return Bluebird.try(function () {\n      // If we don't have bitgoFeeInfo, or address is already set, don't get a new one\n      if (!bitgoFeeInfo || bitgoFeeInfo.address) {\n        return;\n      }\n      return bitgo.getBitGoFeeAddress().then(function (result) {\n        bitgoFeeInfo.address = result.address;\n      });\n    });\n  };\n\n  // Get a dynamic fee estimate from the BitGo server if feeTxConfirmTarget\n  // is specified or if no fee-related params are specified\n  const getDynamicFeeRateEstimate = function () {\n    if (params.feeTxConfirmTarget || !feeParamsDefined) {\n      return bitgo\n        .estimateFee({\n          numBlocks: params.feeTxConfirmTarget,\n          maxFee: params.maxFeeRate,\n          inputs: zeroConfUnspentTxIds,\n          txSize: estTxSize,\n          cpfpAware: true,\n        })\n        .then(function (result) {\n          const estimatedFeeRate = result.cpfpFeePerKb;\n          const minimum = params.instant\n            ? Math.max(constants.minFeeRate, constants.minInstantFeeRate)\n            : constants.minFeeRate;\n          // 5 satoshis per byte\n          // it is worth noting that the padding only applies when the threshold is crossed, but not when the delta is less than the padding\n          const padding = 5000;\n          if (estimatedFeeRate < minimum) {\n            console.log(\n              new Date() +\n                ': Error when estimating fee for send from ' +\n                params.wallet.id() +\n                ', it was too low - ' +\n                estimatedFeeRate\n            );\n            feeRate = minimum + padding;\n          } else if (estimatedFeeRate > params.maxFeeRate) {\n            feeRate = params.maxFeeRate - padding;\n          } else {\n            feeRate = estimatedFeeRate;\n          }\n          return feeRate;\n        })\n        .catch(function (e) {\n          // sanity check failed on tx size\n          if (_.includes(e.message, 'invalid txSize')) {\n            return Bluebird.reject(e);\n          } else {\n            // couldn't estimate the fee, proceed using the default\n            feeRate = constants.fallbackFeeRate;\n            console.log('Error estimating fee for send from ' + params.wallet.id() + ': ' + e.message);\n            return Bluebird.resolve();\n          }\n        });\n    }\n  };\n\n  // Get the unspents for the sending wallet.\n  const getUnspents = function () {\n    if (params.unspents) {\n      // we just wanna use custom unspents\n      unspents = params.unspents;\n      return;\n    }\n\n    // Get enough unspents for the requested amount\n    const options = _.merge({}, params.unspentsFetchParams || {}, {\n      target: totalAmount,\n      minSize: params.minUnspentSize || 0,\n      instant: params.instant, // insist on instant unspents only\n      targetWalletUnspents: params.targetWalletUnspents,\n    });\n    if (params.instant) {\n      options.instant = params.instant; // insist on instant unspents only\n    }\n\n    return params.wallet.unspentsPaged(options).then(function (results) {\n      console.log(`Unspents fetched\\n:  ${JSON.stringify(results, null, 2)}`);\n      totalUnspentsCount = results.total;\n      fetchedUnspentsCount = results.count;\n      unspents = results.unspents.filter(function (u) {\n        const confirms = u.confirmations || 0;\n        if (!params.enforceMinConfirmsForChange && u.isChange) {\n          return true;\n        }\n        return confirms >= minConfirms;\n      });\n\n      // abort early if there's no viable unspents, because it won't be possible to create the txn later\n      if (unspents.length === 0) {\n        throw Error('0 unspents available for transaction creation');\n      }\n\n      // create array of unconfirmed unspent ID strings of the form \"txHash:outputIndex\"\n      zeroConfUnspentTxIds = _(results.unspents)\n        .filter(function (u) {\n          return !u.confirmations;\n        })\n        .map(function (u) {\n          return u.tx_hash + ':' + u.tx_output_n;\n        })\n        .value();\n      if (_.isEmpty(zeroConfUnspentTxIds)) {\n        // we don't want to pass an empty array of inputs to the server, because it assumes if the\n        // inputs arguments exists, it contains values\n        zeroConfUnspentTxIds = undefined;\n      }\n\n      // For backwards compatibility, respect the old splitChangeSize=0 parameter\n      if (!params.noSplitChange && params.splitChangeSize !== 0) {\n        extraChangeAmounts = results.extraChangeAmounts || [];\n      }\n    });\n  };\n\n  // Get the unspents for the single key fee address\n  let feeSingleKeyUnspents: BitGoUnspent[] = [];\n  const getUnspentsForSingleKey = function () {\n    if (feeSingleKeySourceAddress) {\n      let feeTarget = 0.01e8;\n      if (params.instant) {\n        feeTarget += totalAmount * 0.001;\n      }\n      return bitgo\n        .get(bitgo.url('/address/' + feeSingleKeySourceAddress + '/unspents?target=' + feeTarget))\n        .then(function (response) {\n          if (response.body.total <= 0) {\n            throw new Error('No unspents available in single key fee source');\n          }\n          feeSingleKeyUnspents = response.body.unspents;\n        });\n    }\n  };\n\n  let minerFeeInfo: any = {};\n  let txInfo: any = {};\n\n  // Iterate unspents, sum the inputs, and save _inputs with the total\n  // input amount and final list of inputs to use with the transaction.\n  let feeSingleKeyUnspentsUsed: BitGoUnspent[] = [];\n\n  const collectInputs = function () {\n    if (!unspents.length) {\n      throw new Error('no unspents available on wallet');\n    }\n    inputAmount = 0;\n\n    // Calculate the cost of spending a single input, i.e. the smallest economical unspent value\n    return Bluebird.try(function () {\n      if (_.isNumber(params.feeRate) || _.isNumber(params.originalFeeRate)) {\n        return !_.isUndefined(params.feeRate) ? params.feeRate : params.originalFeeRate;\n      } else {\n        return bitgo\n          .estimateFee({\n            numBlocks: params.feeTxConfirmTarget,\n            maxFee: params.maxFeeRate,\n          })\n          .then(function (feeRateEstimate) {\n            return feeRateEstimate.feePerKb;\n          });\n      }\n    })\n      .then(function (feeRate) {\n        // Don't spend inputs that cannot pay for their own cost.\n        let minInputValue = 0;\n        if (_.isInteger(params.minUnspentSize)) {\n          minInputValue = params.minUnspentSize;\n        }\n\n        let prunedUnspentCount = 0;\n        const originalUnspentCount = unspents.length;\n        unspents = _.filter(unspents, function (unspent) {\n          const isSegwitInput = !!unspent.witnessScript;\n          const currentInputSize = isSegwitInput ? VirtualSizes.txP2shP2wshInputSize : VirtualSizes.txP2shInputSize;\n          const feeBasedMinInputValue = (feeRate * currentInputSize) / 1000;\n          const currentMinInputValue = Math.max(minInputValue, feeBasedMinInputValue);\n          if (currentMinInputValue > unspent.value) {\n            // pruning unspent\n            const pruneDetails = {\n              generalMinInputValue: minInputValue,\n              feeBasedMinInputValue,\n              currentMinInputValue,\n              feeRate,\n              inputSize: currentInputSize,\n              unspent: unspent,\n            };\n            debug(`pruning unspent: ${JSON.stringify(pruneDetails, null, 4)}`);\n            prunedUnspentCount++;\n            return false;\n          }\n          return true;\n        });\n\n        if (prunedUnspentCount > 0) {\n          debug(`pruned ${prunedUnspentCount} out of ${originalUnspentCount} unspents`);\n        }\n\n        if (unspents.length === 0) {\n          throw new Error('insufficient funds');\n        }\n        let segwitInputCount = 0;\n        unspents.every(function (unspent) {\n          if (unspent.witnessScript) {\n            segwitInputCount++;\n          }\n          inputAmount += unspent.value;\n          transaction.addInput(unspent.tx_hash, unspent.tx_output_n, 0xffffffff);\n\n          return inputAmount < (feeSingleKeySourceAddress ? totalOutputAmount : totalAmount);\n        });\n\n        // if paying fees from an external single key wallet, add the inputs\n        if (feeSingleKeySourceAddress) {\n          // collect the amount used in the fee inputs so we can get change later\n          feeSingleKeyInputAmount = 0;\n          feeSingleKeyUnspentsUsed = [];\n          feeSingleKeyUnspents.every(function (unspent) {\n            feeSingleKeyInputAmount += unspent.value;\n            inputAmount += unspent.value;\n            transaction.addInput(unspent.tx_hash, unspent.tx_output_n);\n            feeSingleKeyUnspentsUsed.push(unspent);\n            // use the fee wallet to pay miner fees and potentially instant fees\n            return feeSingleKeyInputAmount < fee + (bitgoFeeInfo ? bitgoFeeInfo.amount : 0);\n          });\n        }\n\n        txInfo = {\n          nP2shInputs: transaction.tx.ins.length - (feeSingleKeySourceAddress ? 1 : 0) - segwitInputCount,\n          nP2shP2wshInputs: segwitInputCount,\n          nP2pkhInputs: feeSingleKeySourceAddress ? 1 : 0,\n          // add single key source address change\n          nOutputs:\n            recipients.length +\n            1 + // recipients and change\n            extraChangeAmounts.length + // extra change splitting\n            (bitgoFeeInfo && bitgoFeeInfo.amount > 0 ? 1 : 0) + // add output for bitgo fee\n            (feeSingleKeySourceAddress ? 1 : 0),\n        };\n\n        // As per the response of get unspents API, for v1 safe wallets redeemScript is returned\n        // in the response in hex format\n        containsUncompressedPublicKeys = unspents.some(\n          (u) => u.redeemScript.length === 201 * 2 /* hex length is twice the length in bytes */\n        );\n\n        estTxSize = estimateTransactionSize({\n          containsUncompressedPublicKeys,\n          nP2shInputs: txInfo.nP2shInputs,\n          nP2shP2wshInputs: txInfo.nP2shP2wshInputs,\n          nP2pkhInputs: txInfo.nP2pkhInputs,\n          nOutputs: txInfo.nOutputs,\n        });\n      })\n      .then(getDynamicFeeRateEstimate)\n      .then(function () {\n        minerFeeInfo = exports.calculateMinerFeeInfo({\n          bitgo: params.wallet.bitgo,\n          containsUncompressedPublicKeys,\n          feeRate: feeRate,\n          nP2shInputs: txInfo.nP2shInputs,\n          nP2shP2wshInputs: txInfo.nP2shP2wshInputs,\n          nP2pkhInputs: txInfo.nP2pkhInputs,\n          nOutputs: txInfo.nOutputs,\n        });\n\n        if (shouldComputeBestFee) {\n          const approximateFee = minerFeeInfo.fee;\n          const shouldRecurse = _.isUndefined(fee) || approximateFee > fee;\n          fee = approximateFee;\n          // Recompute totalAmount from scratch\n          totalAmount = fee + totalOutputAmount;\n          if (bitgoFeeInfo) {\n            totalAmount += bitgoFeeInfo.amount;\n          }\n          if (shouldRecurse) {\n            // if fee changed, re-collect inputs\n            inputAmount = 0;\n            transaction = utxolib.bitgo.createTransactionBuilderForNetwork(network);\n            return collectInputs();\n          }\n        }\n\n        const totalFee = fee + (bitgoFeeInfo ? bitgoFeeInfo.amount : 0);\n\n        if (feeSingleKeySourceAddress) {\n          const summedSingleKeyUnspents = _.sumBy(feeSingleKeyUnspents, 'value');\n          if (totalFee > summedSingleKeyUnspents) {\n            const err: any = new Error(\n              'Insufficient fee amount available in single key fee source: ' + summedSingleKeyUnspents\n            );\n            err.result = {\n              fee: fee,\n              feeRate: feeRate,\n              estimatedSize: minerFeeInfo.size,\n              available: inputAmount,\n              bitgoFee: bitgoFeeInfo,\n              txInfo: txInfo,\n            };\n            return Bluebird.reject(err);\n          }\n        }\n\n        if (inputAmount < (feeSingleKeySourceAddress ? totalOutputAmount : totalAmount)) {\n          // The unspents we're using for inputs do not have sufficient value on them to\n          // satisfy the user's requested spend amount. That may be because the wallet's balance\n          // is simply too low, or it might be that the wallet's balance is sufficient but\n          // we didn't fetch enough unspents. Too few unspents could result from the wallet\n          // having many small unspents and we hit our limit on the number of inputs we can use\n          // in a txn, or it might have been that the filters the user passed in (like minConfirms)\n          // disqualified too many of the unspents\n          let err;\n          if (totalUnspentsCount === fetchedUnspentsCount) {\n            // we fetched every unspent the wallet had, but it still wasn't enough\n            err = new Error('Insufficient funds');\n          } else {\n            // we weren't able to fetch all the unspents on the wallet\n            err = new Error(\n              `Transaction size too large due to too many unspents. Can send only ${inputAmount} satoshis in this transaction`\n            );\n          }\n          err.result = {\n            fee: fee,\n            feeRate: feeRate,\n            estimatedSize: minerFeeInfo.size,\n            available: inputAmount,\n            bitgoFee: bitgoFeeInfo,\n            txInfo: txInfo,\n          };\n          return Bluebird.reject(err);\n        }\n      });\n  };\n\n  // Add the outputs for this transaction.\n  const collectOutputs = function () {\n    if (minerFeeInfo.size >= 90000) {\n      throw new Error('transaction too large: estimated size ' + minerFeeInfo.size + ' bytes');\n    }\n\n    const outputs: Output[] = [];\n\n    recipients.forEach(function (recipient) {\n      let script;\n      if (_.isString(recipient.address)) {\n        script = utxolib.address.toOutputScript(recipient.address, network);\n      } else if (_.isObject(recipient.script)) {\n        script = recipient.script;\n      } else {\n        throw new Error('neither recipient address nor script was provided');\n      }\n\n      // validate travelInfo if it exists\n      let travelInfo;\n      if (!_.isEmpty(recipient.travelInfo)) {\n        travelInfo = recipient.travelInfo;\n        // Better to avoid trouble now, before tx is created\n        bitgo.travelRule().validateTravelInfo(travelInfo);\n      }\n\n      outputs.push({\n        script: script,\n        amount: recipient.amount,\n        travelInfo: travelInfo,\n      });\n    });\n\n    opReturns.forEach(function ({ message, amount }) {\n      const script = utxolib.script.fromASM('OP_RETURN ' + Buffer.from(message).toString('hex'));\n      outputs.push({ script, amount });\n    });\n\n    const getChangeOutputs = function (changeAmount: number): Output[] | Bluebird<Output[]> {\n      if (changeAmount < 0) {\n        throw new Error('negative change amount: ' + changeAmount);\n      }\n\n      const result: Output[] = [];\n      // if we paid fees from a single key wallet, return the fee change first\n      if (feeSingleKeySourceAddress) {\n        const feeSingleKeyWalletChangeAmount =\n          feeSingleKeyInputAmount - (fee + (bitgoFeeInfo ? bitgoFeeInfo.amount : 0));\n        if (feeSingleKeyWalletChangeAmount >= constants.minOutputSize) {\n          result.push({ address: feeSingleKeySourceAddress, amount: feeSingleKeyWalletChangeAmount });\n          changeAmount = changeAmount - feeSingleKeyWalletChangeAmount;\n        }\n      }\n\n      if (changeAmount < constants.minOutputSize) {\n        // Give it to the miners\n        return result;\n      }\n\n      if (params.wallet.type() === 'safe') {\n        return params.wallet.addresses().then(function (response) {\n          result.push({ address: response.addresses[0].address, amount: changeAmount });\n          return result;\n        });\n      }\n\n      let extraChangeTotal = _.sum(extraChangeAmounts);\n      // Sanity check\n      if (extraChangeTotal > changeAmount) {\n        extraChangeAmounts = [];\n        extraChangeTotal = 0;\n      }\n\n      // copy and add remaining change amount\n      const allChangeAmounts = extraChangeAmounts.slice(0);\n      allChangeAmounts.push(changeAmount - extraChangeTotal);\n\n      // Recursive async func to add all change outputs\n      const addChangeOutputs = function (): Output[] | Bluebird<Output[]> {\n        const thisAmount = allChangeAmounts.shift();\n        if (!thisAmount) {\n          return result;\n        }\n        return Bluebird.try(function () {\n          if (params.changeAddress) {\n            // If user passed a change address, use it for all outputs\n            return params.changeAddress;\n          } else {\n            // Otherwise create a new address per output, for privacy\n            // determine if segwit or not\n            const changeChain = params.wallet.getChangeChain(params);\n            return params.wallet.createAddress({ chain: changeChain, validate: validate }).then(function (result) {\n              return result.address;\n            });\n          }\n        }).then(function (address) {\n          result.push({ address: address, amount: thisAmount });\n          return addChangeOutputs();\n        });\n      };\n\n      return addChangeOutputs();\n    };\n\n    // Add change output(s) and instant fee output if applicable\n    return Bluebird.try(function () {\n      return getChangeOutputs(inputAmount - totalAmount);\n    }).then(function (result) {\n      changeOutputs = result;\n      const extraOutputs = changeOutputs.concat([]); // copy the array\n      if (bitgoFeeInfo && bitgoFeeInfo.amount > 0) {\n        extraOutputs.push(bitgoFeeInfo);\n      }\n      extraOutputs.forEach(function (output) {\n        if ((output as AddressOutput).address) {\n          (output as ScriptOutput).script = utxolib.address.toOutputScript((output as AddressOutput).address, network);\n        }\n\n        // decide where to put the outputs - default is to randomize unless forced to end\n        const outputIndex = params.forceChangeAtEnd ? outputs.length : _.random(0, outputs.length);\n        outputs.splice(outputIndex, 0, output);\n      });\n\n      // Add all outputs to the transaction\n      outputs.forEach(function (output) {\n        transaction.addOutput((output as ScriptOutput).script, output.amount);\n      });\n\n      travelInfos = _(outputs)\n        .map(function (output, index) {\n          const result = output.travelInfo;\n          if (!result) {\n            return undefined;\n          }\n          result.outputIndex = index;\n          return result;\n        })\n        .filter()\n        .value();\n    });\n  };\n\n  // Serialize the transaction, returning what is needed to sign it\n  const serialize = function () {\n    // only need to return the unspents that were used and just the chainPath, redeemScript, and instant flag\n    const pickedUnspents: any = _.map(unspents, function (unspent) {\n      return _.pick(unspent, ['chainPath', 'redeemScript', 'instant', 'witnessScript', 'script', 'value']);\n    });\n    const prunedUnspents = _.slice(pickedUnspents, 0, transaction.tx.ins.length - feeSingleKeyUnspentsUsed.length);\n    _.each(feeSingleKeyUnspentsUsed, function (feeUnspent) {\n      prunedUnspents.push({ redeemScript: false, chainPath: false }); // mark as false to signify a non-multisig address\n    });\n    const result: any = {\n      transactionHex: transaction.buildIncomplete().toHex(),\n      unspents: prunedUnspents,\n      fee: fee,\n      changeAddresses: changeOutputs.map(function (co) {\n        return _.pick(co, ['address', 'path', 'amount']);\n      }),\n      walletId: params.wallet.id(),\n      walletKeychains: params.wallet.keychains,\n      feeRate: feeRate,\n      instant: params.instant,\n      bitgoFee: bitgoFeeInfo,\n      estimatedSize: minerFeeInfo.size,\n      txInfo: txInfo,\n      travelInfos: travelInfos,\n    };\n\n    // Add for backwards compatibility\n    if (result.instant && bitgoFeeInfo) {\n      result.instantFee = _.pick(bitgoFeeInfo, ['amount', 'address']);\n    }\n\n    return result;\n  };\n\n  return Bluebird.try(function () {\n    return getBitGoFee();\n  })\n    .then(function () {\n      return Bluebird.all([getBitGoFeeAddress(), getUnspents(), getUnspentsForSingleKey()]);\n    })\n    .then(collectInputs)\n    .then(collectOutputs)\n    .then(serialize);\n};\n\n/**\n * Estimate the size of a transaction in bytes based on the number of\n * inputs and outputs present.\n * @params params {\n *   nP2shInputs: number of P2SH (multisig) inputs\n *   nP2pkhInputs: number of P2PKH (single sig) inputs\n *   nOutputs: number of outputs\n * }\n *\n * @returns size: estimated size of the transaction in bytes\n */\nconst estimateTransactionSize = function (params) {\n  if (!_.isInteger(params.nP2shInputs) || params.nP2shInputs < 0) {\n    throw new Error('expecting positive nP2shInputs');\n  }\n  if (!_.isInteger(params.nP2pkhInputs) || params.nP2pkhInputs < 0) {\n    throw new Error('expecting positive nP2pkhInputs to be numeric');\n  }\n  if (!_.isInteger(params.nP2shP2wshInputs) || params.nP2shP2wshInputs < 0) {\n    throw new Error('expecting positive nP2shP2wshInputs to be numeric');\n  }\n  if (params.nP2shInputs + params.nP2shP2wshInputs < 1) {\n    throw new Error('expecting at least one nP2shInputs or nP2shP2wshInputs');\n  }\n  if (!_.isInteger(params.nOutputs) || params.nOutputs < 1) {\n    throw new Error('expecting positive nOutputs');\n  }\n\n  // The size of an uncompressed public key is 32 bytes more than the compressed key,\n  // and hence, needs to be accounted for in the transaction size estimation.\n  const uncompressedPublicKeysTripleCorrectionFactor = 32 * 3;\n\n  return (\n    // This is not quite accurate - if there is a mix of inputs scripts where some used\n    // compressed keys and some used uncompressed keys, we would overestimate the size.\n    // Since we don't have mixed input sets, this should not be an issue in practice.\n    (VirtualSizes.txP2shInputSize +\n      (params.containsUncompressedPublicKeys ? uncompressedPublicKeysTripleCorrectionFactor : 0)) *\n      params.nP2shInputs +\n    VirtualSizes.txP2shP2wshInputSize * (params.nP2shP2wshInputs || 0) +\n    VirtualSizes.txP2pkhInputSizeUncompressedKey * (params.nP2pkhInputs || 0) +\n    VirtualSizes.txP2pkhOutputSize * params.nOutputs +\n    // if the tx contains at least one segwit input, the tx overhead is increased by 1\n    VirtualSizes.txOverheadSize +\n    (params.nP2shP2wshInputs > 0 ? 1 : 0)\n  );\n};\n\n/**\n * Calculate the fee and estimated size in bytes for a transaction.\n * @params params {\n *   bitgo: bitgo object\n *   feeRate: satoshis per kilobyte\n *   nP2shInputs: number of P2SH (multisig) inputs\n *   nP2pkhInputs: number of P2PKH (single sig) inputs\n *   nOutputs: number of outputs\n * }\n *\n * @returns {\n *   size: estimated size of the transaction in bytes\n *   fee: estimated fee in satoshis for the transaction\n *   feeRate: fee rate that was used to estimate the fee for the transaction\n * }\n */\nexports.calculateMinerFeeInfo = function (params) {\n  const feeRateToUse = params.feeRate || params.bitgo.getConstants().fallbackFeeRate;\n  const estimatedSize = estimateTransactionSize(params);\n\n  return {\n    size: estimatedSize,\n    fee: Math.ceil((estimatedSize * feeRateToUse) / 1000),\n    feeRate: feeRateToUse,\n  };\n};\n\n/*\n * Given a transaction hex, unspent information (chain path and redeem scripts), and the keychain xprv,\n * perform key derivation and sign the inputs in the transaction based on the unspent information provided\n *\n * @params:\n *  transactionHex serialized form of the transaction in hex\n *  unspents array of unspent information, where each unspent is a chainPath and redeemScript with the same\n *  index as the inputs in the transactionHex\n *  keychain Keychain containing the xprv to sign with. For legacy support of safe wallets, keychain can\n also be a WIF private key.\n *  signingKey private key in WIF for safe wallets, when keychain is unavailable\n *  validate client-side signature verification - can be disabled for improved performance (signatures\n *           are still validated server-side).\n *  feeSingleKeyWIF Use the address based on this private key to pay fees\n * @returns {*}\n */\nexports.signTransaction = function (params) {\n  let keychain = params.keychain; // duplicate so as to not mutate below\n\n  const validate = params.validate === undefined ? true : params.validate;\n  let privKey;\n  if (!_.isString(params.transactionHex)) {\n    throw new Error('expecting the transaction hex as a string');\n  }\n  if (!Array.isArray(params.unspents)) {\n    throw new Error('expecting the unspents array');\n  }\n  if (!_.isBoolean(validate)) {\n    throw new Error('expecting validate to be a boolean');\n  }\n  let network = getNetwork();\n  const enableBCH = _.isBoolean(params.forceBCH) && params.forceBCH === true;\n\n  if (!_.isObject(keychain) || !_.isString((keychain as any).xprv)) {\n    if (_.isString(params.signingKey)) {\n      privKey = utxolib.ECPair.fromWIF(params.signingKey, network as utxolib.BitcoinJSNetwork);\n      keychain = undefined;\n    } else {\n      throw new Error('expecting the keychain object with xprv');\n    }\n  }\n\n  let feeSingleKey;\n  if (params.feeSingleKeyWIF) {\n    feeSingleKey = utxolib.ECPair.fromWIF(params.feeSingleKeyWIF, network as utxolib.BitcoinJSNetwork);\n  }\n\n  debug('Network: %O', network);\n\n  if (enableBCH) {\n    debug('Enabling BCH…');\n    network = utxolib.networks.bitcoincash;\n    debug('New network: %O', network);\n  }\n\n  const transaction = utxolib.bitgo.createTransactionFromHex(params.transactionHex, network);\n  if (transaction.ins.length !== params.unspents.length) {\n    throw new Error('length of unspents array should equal to the number of transaction inputs');\n  }\n\n  // decorate transaction with input values for TransactionBuilder instantiation\n  const isUtxoTx = _.isObject(transaction) && Array.isArray((transaction as any).ins);\n  const areValidUnspents = _.isObject(params) && Array.isArray((params as any).unspents);\n  if (isUtxoTx && areValidUnspents) {\n    // extend the transaction inputs with the values\n    const inputValues = _.map((params as any).unspents, (u) => _.pick(u, 'value'));\n    transaction.ins.map((currentItem, index) => _.extend(currentItem, inputValues[index]));\n  }\n\n  let rootExtKey;\n  if (keychain) {\n    rootExtKey = bip32.fromBase58(keychain.xprv);\n  }\n\n  const txb = utxolib.bitgo.createTransactionBuilderFromTransaction(transaction);\n\n  for (let index = 0; index < txb.tx.ins.length; ++index) {\n    const currentUnspent = params.unspents[index];\n    if (currentUnspent.redeemScript === false) {\n      // this is the input from a single key fee address\n      if (!feeSingleKey) {\n        throw new Error('single key address used in input but feeSingleKeyWIF not provided');\n      }\n\n      if (enableBCH) {\n        feeSingleKey.network = network;\n      }\n\n      txb.sign(index, feeSingleKey);\n      continue;\n    }\n\n    if (currentUnspent.witnessScript && enableBCH) {\n      throw new Error('BCH does not support segwit inputs');\n    }\n\n    const chainPath = currentUnspent.chainPath;\n    if (rootExtKey) {\n      const { walletSubPath = '/0/0' } = keychain;\n      const path = sanitizeLegacyPath(keychain.path + walletSubPath + chainPath);\n      debug(\n        'derived user key path \"%s\" using keychain path \"%s\", walletSubPath \"%s\", keychain walletSubPath \"%s\" and chainPath \"%s\"',\n        path,\n        keychain.path,\n        walletSubPath,\n        keychain.walletSubPath,\n        chainPath\n      );\n      privKey = rootExtKey.derivePath(path);\n    }\n\n    privKey.network = network;\n\n    // subscript is the part of the output script after the OP_CODESEPARATOR.\n    // Since we are only ever signing p2sh outputs, which do not have\n    // OP_CODESEPARATORS, it is always the output script.\n    const subscript = Buffer.from(currentUnspent.redeemScript, 'hex');\n    currentUnspent.validationScript = subscript;\n\n    // In order to sign with bitcoinjs-lib, we must use its transaction\n    // builder, confusingly named the same exact thing as our transaction\n    // builder, but with inequivalent behavior.\n    try {\n      const witnessScript = currentUnspent.witnessScript ? Buffer.from(currentUnspent.witnessScript, 'hex') : undefined;\n      const sigHash = utxolib.bitgo.getDefaultSigHash(network);\n      txb.sign(index, privKey, subscript, sigHash, currentUnspent.value, witnessScript);\n      debug(`Signed transaction input ${index}`);\n    } catch (e) {\n      // try fallback derivation path (see BG-46497)\n      let fallbackSigningSuccessful = false;\n      try {\n        const fallbackPath = sanitizeLegacyPath(keychain.path + chainPath);\n        debug(\n          'derived fallback user key path \"%s\" using keychain path \"%s\" and chainPath \"%s\"',\n          fallbackPath,\n          keychain.path,\n          chainPath\n        );\n        privKey = rootExtKey.derivePath(fallbackPath);\n        const witnessScript = currentUnspent.witnessScript\n          ? Buffer.from(currentUnspent.witnessScript, 'hex')\n          : undefined;\n        const sigHash = utxolib.bitgo.getDefaultSigHash(network);\n        txb.sign(index, privKey, subscript, sigHash, currentUnspent.value, witnessScript);\n        fallbackSigningSuccessful = true;\n      } catch (fallbackError) {\n        debug('input sign failed for fallback path: %s', fallbackError.message);\n      }\n      // we need to know what's causing this\n      if (!fallbackSigningSuccessful) {\n        e.result = {\n          unspent: currentUnspent,\n        };\n        e.message = `Failed to sign input #${index} - ${e.message} - ${JSON.stringify(e.result, null, 4)} - \\n${\n          e.stack\n        }`;\n        debug('input sign failed: %s', e.message);\n        return Bluebird.reject(e);\n      }\n    }\n  }\n\n  const partialTransaction = txb.buildIncomplete();\n\n  if (validate) {\n    partialTransaction.ins.forEach((input, index) => {\n      const signatureCount = utxolib.bitgo\n        .getSignatureVerifications(partialTransaction, index, params.unspents[index].value)\n        .filter((v) => v.signedBy !== undefined).length;\n      debug(`Signature count for input ${index}: ${signatureCount}`);\n      if (signatureCount < 1) {\n        throw new Error('expected at least one valid signature');\n      }\n      if (params.fullLocalSigning && signatureCount < 2) {\n        throw new Error('fullLocalSigning set: expected at least two valid signatures');\n      }\n    });\n  }\n\n  return Bluebird.resolve({\n    transactionHex: partialTransaction.toHex(),\n  });\n};\n"]}
|