@bitgo-beta/abstract-utxo 1.6.1-alpha.6 → 1.6.1-alpha.61
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 +154 -0
- package/dist/src/abstractUtxoCoin.d.ts +88 -14
- package/dist/src/abstractUtxoCoin.d.ts.map +1 -1
- package/dist/src/abstractUtxoCoin.js +204 -139
- package/dist/src/parseOutput.d.ts.map +1 -1
- package/dist/src/parseOutput.js +12 -1
- package/dist/src/recovery/RecoveryProvider.d.ts +6 -5
- package/dist/src/recovery/RecoveryProvider.d.ts.map +1 -1
- package/dist/src/recovery/RecoveryProvider.js +1 -1
- package/dist/src/recovery/backupKeyRecovery.d.ts +6 -3
- package/dist/src/recovery/backupKeyRecovery.d.ts.map +1 -1
- package/dist/src/recovery/backupKeyRecovery.js +4 -2
- package/dist/src/recovery/crossChainRecovery.d.ts +11 -23
- package/dist/src/recovery/crossChainRecovery.d.ts.map +1 -1
- package/dist/src/recovery/crossChainRecovery.js +19 -65
- package/dist/src/sign.d.ts +26 -3
- package/dist/src/sign.d.ts.map +1 -1
- package/dist/src/sign.js +68 -2
- package/dist/src/transaction.d.ts +36 -0
- package/dist/src/transaction.d.ts.map +1 -0
- package/dist/src/transaction.js +205 -0
- package/dist/tsconfig.tsbuildinfo +1 -7734
- package/package.json +8 -8
|
@@ -0,0 +1,205 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.explainTx = exports.explainPsbt = exports.getTxInputs = exports.getPsbtTxInputs = void 0;
|
|
4
|
+
const utxolib = require("@bitgo-beta/utxo-lib");
|
|
5
|
+
const utxo_lib_1 = require("@bitgo-beta/utxo-lib");
|
|
6
|
+
/**
|
|
7
|
+
* Get the inputs for a psbt from a prebuild.
|
|
8
|
+
*/
|
|
9
|
+
function getPsbtTxInputs(psbtArg, network) {
|
|
10
|
+
const psbt = psbtArg instanceof utxolib.bitgo.UtxoPsbt ? psbtArg : utxolib.bitgo.createPsbtFromHex(psbtArg, network);
|
|
11
|
+
const txInputs = psbt.txInputs;
|
|
12
|
+
return psbt.data.inputs.map((input, index) => {
|
|
13
|
+
let address;
|
|
14
|
+
let value;
|
|
15
|
+
if (input.witnessUtxo) {
|
|
16
|
+
address = utxolib.address.fromOutputScript(input.witnessUtxo.script, network);
|
|
17
|
+
value = input.witnessUtxo.value;
|
|
18
|
+
}
|
|
19
|
+
else if (input.nonWitnessUtxo) {
|
|
20
|
+
const tx = utxolib.bitgo.createTransactionFromBuffer(input.nonWitnessUtxo, network, {
|
|
21
|
+
amountType: 'bigint',
|
|
22
|
+
});
|
|
23
|
+
const txId = Buffer.from(txInputs[index].hash).reverse().toString('hex');
|
|
24
|
+
if (tx.getId() !== txId) {
|
|
25
|
+
throw new Error('input transaction hex does not match id');
|
|
26
|
+
}
|
|
27
|
+
const prevTxOutputIndex = txInputs[index].index;
|
|
28
|
+
address = utxolib.address.fromOutputScript(tx.outs[prevTxOutputIndex].script, network);
|
|
29
|
+
value = tx.outs[prevTxOutputIndex].value;
|
|
30
|
+
}
|
|
31
|
+
else {
|
|
32
|
+
throw new Error('psbt input is missing both witnessUtxo and nonWitnessUtxo');
|
|
33
|
+
}
|
|
34
|
+
return { address, value, valueString: value.toString() };
|
|
35
|
+
});
|
|
36
|
+
}
|
|
37
|
+
exports.getPsbtTxInputs = getPsbtTxInputs;
|
|
38
|
+
/**
|
|
39
|
+
* Get the inputs for a transaction from a prebuild.
|
|
40
|
+
*/
|
|
41
|
+
async function getTxInputs(params) {
|
|
42
|
+
const { txPrebuild, bitgo, coin, disableNetworking, reqId } = params;
|
|
43
|
+
if (!txPrebuild.txHex) {
|
|
44
|
+
throw new Error(`txPrebuild.txHex not set`);
|
|
45
|
+
}
|
|
46
|
+
const transaction = coin.createTransactionFromHex(txPrebuild.txHex);
|
|
47
|
+
const transactionCache = {};
|
|
48
|
+
return await Promise.all(transaction.ins.map(async (currentInput) => {
|
|
49
|
+
var _a, _b;
|
|
50
|
+
const transactionId = Buffer.from(currentInput.hash).reverse().toString('hex');
|
|
51
|
+
const txHex = (_b = (_a = txPrebuild.txInfo) === null || _a === void 0 ? void 0 : _a.txHexes) === null || _b === void 0 ? void 0 : _b[transactionId];
|
|
52
|
+
if (txHex) {
|
|
53
|
+
const localTx = coin.createTransactionFromHex(txHex);
|
|
54
|
+
if (localTx.getId() !== transactionId) {
|
|
55
|
+
throw new Error('input transaction hex does not match id');
|
|
56
|
+
}
|
|
57
|
+
const currentOutput = localTx.outs[currentInput.index];
|
|
58
|
+
const address = utxolib.address.fromOutputScript(currentOutput.script, coin.network);
|
|
59
|
+
return {
|
|
60
|
+
address,
|
|
61
|
+
value: currentOutput.value,
|
|
62
|
+
valueString: currentOutput.value.toString(),
|
|
63
|
+
};
|
|
64
|
+
}
|
|
65
|
+
else if (!transactionCache[transactionId]) {
|
|
66
|
+
if (disableNetworking) {
|
|
67
|
+
throw new Error('attempting to retrieve transaction details externally with networking disabled');
|
|
68
|
+
}
|
|
69
|
+
if (reqId) {
|
|
70
|
+
bitgo.setRequestTracer(reqId);
|
|
71
|
+
}
|
|
72
|
+
transactionCache[transactionId] = await bitgo.get(coin.url(`/public/tx/${transactionId}`)).result();
|
|
73
|
+
}
|
|
74
|
+
const transactionDetails = transactionCache[transactionId];
|
|
75
|
+
return transactionDetails.outputs[currentInput.index];
|
|
76
|
+
}));
|
|
77
|
+
}
|
|
78
|
+
exports.getTxInputs = getTxInputs;
|
|
79
|
+
function explainCommon(tx, params, network) {
|
|
80
|
+
var _a;
|
|
81
|
+
const displayOrder = ['id', 'outputAmount', 'changeAmount', 'outputs', 'changeOutputs'];
|
|
82
|
+
let spendAmount = BigInt(0);
|
|
83
|
+
let changeAmount = BigInt(0);
|
|
84
|
+
const changeOutputs = [];
|
|
85
|
+
const outputs = [];
|
|
86
|
+
const { changeAddresses = [] } = (_a = params.txInfo) !== null && _a !== void 0 ? _a : {};
|
|
87
|
+
tx.outs.forEach((currentOutput) => {
|
|
88
|
+
const currentAddress = utxolib.address.fromOutputScript(currentOutput.script, network);
|
|
89
|
+
const currentAmount = BigInt(currentOutput.value);
|
|
90
|
+
if (changeAddresses.includes(currentAddress)) {
|
|
91
|
+
// this is change
|
|
92
|
+
changeAmount += currentAmount;
|
|
93
|
+
changeOutputs.push({
|
|
94
|
+
address: currentAddress,
|
|
95
|
+
amount: currentAmount.toString(),
|
|
96
|
+
});
|
|
97
|
+
return;
|
|
98
|
+
}
|
|
99
|
+
spendAmount += currentAmount;
|
|
100
|
+
outputs.push({
|
|
101
|
+
address: currentAddress,
|
|
102
|
+
amount: currentAmount.toString(),
|
|
103
|
+
});
|
|
104
|
+
});
|
|
105
|
+
const outputDetails = {
|
|
106
|
+
outputAmount: spendAmount.toString(),
|
|
107
|
+
changeAmount: changeAmount.toString(),
|
|
108
|
+
outputs,
|
|
109
|
+
changeOutputs,
|
|
110
|
+
};
|
|
111
|
+
let fee;
|
|
112
|
+
let locktime;
|
|
113
|
+
if (params.feeInfo) {
|
|
114
|
+
displayOrder.push('fee');
|
|
115
|
+
fee = params.feeInfo;
|
|
116
|
+
}
|
|
117
|
+
if (Number.isInteger(tx.locktime) && tx.locktime > 0) {
|
|
118
|
+
displayOrder.push('locktime');
|
|
119
|
+
locktime = tx.locktime;
|
|
120
|
+
}
|
|
121
|
+
return { displayOrder, id: tx.getId(), ...outputDetails, fee, locktime };
|
|
122
|
+
}
|
|
123
|
+
function getRootWalletKeys(params) {
|
|
124
|
+
var _a;
|
|
125
|
+
const keys = (_a = params.pubs) === null || _a === void 0 ? void 0 : _a.map((xpub) => utxo_lib_1.bip32.fromBase58(xpub));
|
|
126
|
+
return keys && keys.length === 3 ? new utxo_lib_1.bitgo.RootWalletKeys(keys) : undefined;
|
|
127
|
+
}
|
|
128
|
+
function getPsbtInputSignaturesCount(psbt, params) {
|
|
129
|
+
const rootWalletKeys = getRootWalletKeys(params);
|
|
130
|
+
return rootWalletKeys
|
|
131
|
+
? utxo_lib_1.bitgo.getSignatureValidationArrayPsbt(psbt, rootWalletKeys).map((sv) => sv[1].filter((v) => v).length)
|
|
132
|
+
: Array(psbt.data.inputs.length).fill(0);
|
|
133
|
+
}
|
|
134
|
+
function getTxInputSignaturesCount(tx, params, network) {
|
|
135
|
+
var _a, _b, _c;
|
|
136
|
+
const prevOutputs = (_b = (_a = params.txInfo) === null || _a === void 0 ? void 0 : _a.unspents) === null || _b === void 0 ? void 0 : _b.map((u) => utxo_lib_1.bitgo.toOutput(u, network));
|
|
137
|
+
const rootWalletKeys = getRootWalletKeys(params);
|
|
138
|
+
const { unspents = [] } = (_c = params.txInfo) !== null && _c !== void 0 ? _c : {};
|
|
139
|
+
// get the number of signatures per input
|
|
140
|
+
return tx.ins.map((input, idx) => {
|
|
141
|
+
if (unspents.length !== tx.ins.length) {
|
|
142
|
+
return 0;
|
|
143
|
+
}
|
|
144
|
+
if (!prevOutputs) {
|
|
145
|
+
throw new Error(`invalid state`);
|
|
146
|
+
}
|
|
147
|
+
if (!rootWalletKeys) {
|
|
148
|
+
// no pub keys or incorrect number of pub keys
|
|
149
|
+
return 0;
|
|
150
|
+
}
|
|
151
|
+
try {
|
|
152
|
+
return utxo_lib_1.bitgo.verifySignatureWithUnspent(tx, idx, unspents, rootWalletKeys).filter((v) => v).length;
|
|
153
|
+
}
|
|
154
|
+
catch (e) {
|
|
155
|
+
// some other error occurred and we can't validate the signatures
|
|
156
|
+
return 0;
|
|
157
|
+
}
|
|
158
|
+
});
|
|
159
|
+
}
|
|
160
|
+
/**
|
|
161
|
+
* Decompose a raw psbt into useful information, such as the total amounts,
|
|
162
|
+
* change amounts, and transaction outputs.
|
|
163
|
+
*/
|
|
164
|
+
function explainPsbt(params, network) {
|
|
165
|
+
const { txHex } = params;
|
|
166
|
+
let psbt;
|
|
167
|
+
try {
|
|
168
|
+
psbt = utxo_lib_1.bitgo.createPsbtFromHex(txHex, network);
|
|
169
|
+
}
|
|
170
|
+
catch (e) {
|
|
171
|
+
throw new Error('failed to parse psbt hex');
|
|
172
|
+
}
|
|
173
|
+
const tx = psbt.getUnsignedTx();
|
|
174
|
+
const common = explainCommon(tx, params, network);
|
|
175
|
+
const inputSignaturesCount = getPsbtInputSignaturesCount(psbt, params);
|
|
176
|
+
return {
|
|
177
|
+
...common,
|
|
178
|
+
inputSignatures: inputSignaturesCount,
|
|
179
|
+
signatures: inputSignaturesCount.reduce((prev, curr) => (curr > prev ? curr : prev), 0),
|
|
180
|
+
};
|
|
181
|
+
}
|
|
182
|
+
exports.explainPsbt = explainPsbt;
|
|
183
|
+
/**
|
|
184
|
+
* Decompose a raw transaction into useful information, such as the total amounts,
|
|
185
|
+
* change amounts, and transaction outputs.
|
|
186
|
+
*/
|
|
187
|
+
function explainTx(params, coin) {
|
|
188
|
+
const { txHex } = params;
|
|
189
|
+
let tx;
|
|
190
|
+
try {
|
|
191
|
+
tx = coin.createTransactionFromHex(txHex);
|
|
192
|
+
}
|
|
193
|
+
catch (e) {
|
|
194
|
+
throw new Error('failed to parse transaction hex');
|
|
195
|
+
}
|
|
196
|
+
const common = explainCommon(tx, params, coin.network);
|
|
197
|
+
const inputSignaturesCount = getTxInputSignaturesCount(tx, params, coin.network);
|
|
198
|
+
return {
|
|
199
|
+
...common,
|
|
200
|
+
inputSignatures: inputSignaturesCount,
|
|
201
|
+
signatures: inputSignaturesCount.reduce((prev, curr) => (curr > prev ? curr : prev), 0),
|
|
202
|
+
};
|
|
203
|
+
}
|
|
204
|
+
exports.explainTx = explainTx;
|
|
205
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"transaction.js","sourceRoot":"","sources":["../../src/transaction.ts"],"names":[],"mappings":";;;AAAA,gDAAgD;AAShD,mDAAoE;AAEpE;;GAEG;AACH,SAAgB,eAAe,CAC7B,OAAwC,EACxC,OAAwB;IAExB,MAAM,IAAI,GAAG,OAAO,YAAY,OAAO,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,OAAO,CAAC,KAAK,CAAC,iBAAiB,CAAC,OAAO,EAAE,OAAO,CAAC,CAAC;IACrH,MAAM,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC;IAC/B,OAAO,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,KAAK,EAAE,KAAK,EAAE,EAAE;QAC3C,IAAI,OAAe,CAAC;QACpB,IAAI,KAAa,CAAC;QAClB,IAAI,KAAK,CAAC,WAAW,EAAE;YACrB,OAAO,GAAG,OAAO,CAAC,OAAO,CAAC,gBAAgB,CAAC,KAAK,CAAC,WAAW,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;YAC9E,KAAK,GAAG,KAAK,CAAC,WAAW,CAAC,KAAK,CAAC;SACjC;aAAM,IAAI,KAAK,CAAC,cAAc,EAAE;YAC/B,MAAM,EAAE,GAAG,OAAO,CAAC,KAAK,CAAC,2BAA2B,CAAS,KAAK,CAAC,cAAc,EAAE,OAAO,EAAE;gBAC1F,UAAU,EAAE,QAAQ;aACrB,CAAC,CAAC;YACH,MAAM,IAAI,GAAI,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,CAAC,OAAO,EAAa,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;YACrF,IAAI,EAAE,CAAC,KAAK,EAAE,KAAK,IAAI,EAAE;gBACvB,MAAM,IAAI,KAAK,CAAC,yCAAyC,CAAC,CAAC;aAC5D;YACD,MAAM,iBAAiB,GAAG,QAAQ,CAAC,KAAK,CAAC,CAAC,KAAK,CAAC;YAChD,OAAO,GAAG,OAAO,CAAC,OAAO,CAAC,gBAAgB,CAAC,EAAE,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;YACvF,KAAK,GAAG,EAAE,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC,KAAK,CAAC;SAC1C;aAAM;YACL,MAAM,IAAI,KAAK,CAAC,2DAA2D,CAAC,CAAC;SAC9E;QACD,OAAO,EAAE,OAAO,EAAE,KAAK,EAAE,WAAW,EAAE,KAAK,CAAC,QAAQ,EAAE,EAAE,CAAC;IAC3D,CAAC,CAAC,CAAC;AACL,CAAC;AA5BD,0CA4BC;AAED;;GAEG;AACI,KAAK,UAAU,WAAW,CAAkC,MAMlE;IACC,MAAM,EAAE,UAAU,EAAE,KAAK,EAAE,IAAI,EAAE,iBAAiB,EAAE,KAAK,EAAE,GAAG,MAAM,CAAC;IACrE,IAAI,CAAC,UAAU,CAAC,KAAK,EAAE;QACrB,MAAM,IAAI,KAAK,CAAC,0BAA0B,CAAC,CAAC;KAC7C;IACD,MAAM,WAAW,GAAG,IAAI,CAAC,wBAAwB,CAAU,UAAU,CAAC,KAAK,CAAC,CAAC;IAC7E,MAAM,gBAAgB,GAAG,EAAE,CAAC;IAC5B,OAAO,MAAM,OAAO,CAAC,GAAG,CACtB,WAAW,CAAC,GAAG,CAAC,GAAG,CAAC,KAAK,EAAE,YAAY,EAAqE,EAAE;;QAC5G,MAAM,aAAa,GAAI,MAAM,CAAC,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,CAAC,OAAO,EAAa,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;QAC3F,MAAM,KAAK,GAAG,MAAA,MAAA,UAAU,CAAC,MAAM,0CAAE,OAAO,0CAAG,aAAa,CAAC,CAAC;QAC1D,IAAI,KAAK,EAAE;YACT,MAAM,OAAO,GAAG,IAAI,CAAC,wBAAwB,CAAU,KAAK,CAAC,CAAC;YAC9D,IAAI,OAAO,CAAC,KAAK,EAAE,KAAK,aAAa,EAAE;gBACrC,MAAM,IAAI,KAAK,CAAC,yCAAyC,CAAC,CAAC;aAC5D;YACD,MAAM,aAAa,GAAG,OAAO,CAAC,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC,CAAC;YACvD,MAAM,OAAO,GAAG,OAAO,CAAC,OAAO,CAAC,gBAAgB,CAAC,aAAa,CAAC,MAAM,EAAE,IAAI,CAAC,OAAO,CAAC,CAAC;YACrF,OAAO;gBACL,OAAO;gBACP,KAAK,EAAE,aAAa,CAAC,KAAK;gBAC1B,WAAW,EAAE,aAAa,CAAC,KAAK,CAAC,QAAQ,EAAE;aAC5C,CAAC;SACH;aAAM,IAAI,CAAC,gBAAgB,CAAC,aAAa,CAAC,EAAE;YAC3C,IAAI,iBAAiB,EAAE;gBACrB,MAAM,IAAI,KAAK,CAAC,gFAAgF,CAAC,CAAC;aACnG;YACD,IAAI,KAAK,EAAE;gBACT,KAAK,CAAC,gBAAgB,CAAC,KAAK,CAAC,CAAC;aAC/B;YACD,gBAAgB,CAAC,aAAa,CAAC,GAAG,MAAM,KAAK,CAAC,GAAG,CAAC,IAAI,CAAC,GAAG,CAAC,cAAc,aAAa,EAAE,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC;SACrG;QACD,MAAM,kBAAkB,GAAG,gBAAgB,CAAC,aAAa,CAAC,CAAC;QAC3D,OAAO,kBAAkB,CAAC,OAAO,CAAC,YAAY,CAAC,KAAK,CAAC,CAAC;IACxD,CAAC,CAAC,CACH,CAAC;AACJ,CAAC;AA1CD,kCA0CC;AAED,SAAS,aAAa,CACpB,EAAkC,EAClC,MAA0C,EAC1C,OAAwB;;IAExB,MAAM,YAAY,GAAG,CAAC,IAAI,EAAE,cAAc,EAAE,cAAc,EAAE,SAAS,EAAE,eAAe,CAAC,CAAC;IACxF,IAAI,WAAW,GAAG,MAAM,CAAC,CAAC,CAAC,CAAC;IAC5B,IAAI,YAAY,GAAG,MAAM,CAAC,CAAC,CAAC,CAAC;IAC7B,MAAM,aAAa,GAAa,EAAE,CAAC;IACnC,MAAM,OAAO,GAAa,EAAE,CAAC;IAE7B,MAAM,EAAE,eAAe,GAAG,EAAE,EAAE,GAAG,MAAA,MAAM,CAAC,MAAM,mCAAI,EAAE,CAAC;IAErD,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,aAAa,EAAE,EAAE;QAChC,MAAM,cAAc,GAAG,OAAO,CAAC,OAAO,CAAC,gBAAgB,CAAC,aAAa,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;QACvF,MAAM,aAAa,GAAG,MAAM,CAAC,aAAa,CAAC,KAAK,CAAC,CAAC;QAElD,IAAI,eAAe,CAAC,QAAQ,CAAC,cAAc,CAAC,EAAE;YAC5C,iBAAiB;YACjB,YAAY,IAAI,aAAa,CAAC;YAC9B,aAAa,CAAC,IAAI,CAAC;gBACjB,OAAO,EAAE,cAAc;gBACvB,MAAM,EAAE,aAAa,CAAC,QAAQ,EAAE;aACjC,CAAC,CAAC;YACH,OAAO;SACR;QAED,WAAW,IAAI,aAAa,CAAC;QAC7B,OAAO,CAAC,IAAI,CAAC;YACX,OAAO,EAAE,cAAc;YACvB,MAAM,EAAE,aAAa,CAAC,QAAQ,EAAE;SACjC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,MAAM,aAAa,GAAG;QACpB,YAAY,EAAE,WAAW,CAAC,QAAQ,EAAE;QACpC,YAAY,EAAE,YAAY,CAAC,QAAQ,EAAE;QACrC,OAAO;QACP,aAAa;KACd,CAAC;IAEF,IAAI,GAAuB,CAAC;IAC5B,IAAI,QAA4B,CAAC;IAEjC,IAAI,MAAM,CAAC,OAAO,EAAE;QAClB,YAAY,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QACzB,GAAG,GAAG,MAAM,CAAC,OAAO,CAAC;KACtB;IAED,IAAI,MAAM,CAAC,SAAS,CAAC,EAAE,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC,QAAQ,GAAG,CAAC,EAAE;QACpD,YAAY,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;QAC9B,QAAQ,GAAG,EAAE,CAAC,QAAQ,CAAC;KACxB;IAED,OAAO,EAAE,YAAY,EAAE,EAAE,EAAE,EAAE,CAAC,KAAK,EAAE,EAAE,GAAG,aAAa,EAAE,GAAG,EAAE,QAAQ,EAAE,CAAC;AAC3E,CAAC;AAED,SAAS,iBAAiB,CAAkC,MAA0C;;IACpG,MAAM,IAAI,GAAG,MAAA,MAAM,CAAC,IAAI,0CAAE,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,gBAAK,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC,CAAC;IAChE,OAAO,IAAI,IAAI,IAAI,CAAC,MAAM,KAAK,CAAC,CAAC,CAAC,CAAC,IAAI,gBAAK,CAAC,cAAc,CAAC,IAA8B,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC;AAC1G,CAAC;AAED,SAAS,2BAA2B,CAClC,IAAoB,EACpB,MAA0C;IAE1C,MAAM,cAAc,GAAG,iBAAiB,CAAC,MAAM,CAAC,CAAC;IACjD,OAAO,cAAc;QACnB,CAAC,CAAC,gBAAK,CAAC,+BAA+B,CAAC,IAAI,EAAE,cAAc,CAAC,CAAC,GAAG,CAAC,CAAC,EAAE,EAAE,EAAE,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC;QACxG,CAAC,CAAE,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,MAAM,CAAc,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;AAC3D,CAAC;AAED,SAAS,yBAAyB,CAChC,EAAkC,EAClC,MAA0C,EAC1C,OAAwB;;IAExB,MAAM,WAAW,GAAG,MAAA,MAAA,MAAM,CAAC,MAAM,0CAAE,QAAQ,0CAAE,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,gBAAK,CAAC,QAAQ,CAAU,CAAC,EAAE,OAAO,CAAC,CAAC,CAAC;IAC7F,MAAM,cAAc,GAAG,iBAAiB,CAAC,MAAM,CAAC,CAAC;IACjD,MAAM,EAAE,QAAQ,GAAG,EAAE,EAAE,GAAG,MAAA,MAAM,CAAC,MAAM,mCAAI,EAAE,CAAC;IAE9C,yCAAyC;IACzC,OAAO,EAAE,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,KAAK,EAAE,GAAG,EAAU,EAAE;QACvC,IAAI,QAAQ,CAAC,MAAM,KAAK,EAAE,CAAC,GAAG,CAAC,MAAM,EAAE;YACrC,OAAO,CAAC,CAAC;SACV;QACD,IAAI,CAAC,WAAW,EAAE;YAChB,MAAM,IAAI,KAAK,CAAC,eAAe,CAAC,CAAC;SAClC;QACD,IAAI,CAAC,cAAc,EAAE;YACnB,8CAA8C;YAC9C,OAAO,CAAC,CAAC;SACV;QACD,IAAI;YACF,OAAO,gBAAK,CAAC,0BAA0B,CAAU,EAAE,EAAE,GAAG,EAAE,QAAQ,EAAE,cAAc,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC;SAC7G;QAAC,OAAO,CAAC,EAAE;YACV,iEAAiE;YACjE,OAAO,CAAC,CAAC;SACV;IACH,CAAC,CAAC,CAAC;AACL,CAAC;AAED;;;GAGG;AACH,SAAgB,WAAW,CACzB,MAA0C,EAC1C,OAAwB;IAExB,MAAM,EAAE,KAAK,EAAE,GAAG,MAAM,CAAC;IACzB,IAAI,IAAoB,CAAC;IACzB,IAAI;QACF,IAAI,GAAG,gBAAK,CAAC,iBAAiB,CAAC,KAAK,EAAE,OAAO,CAAC,CAAC;KAChD;IAAC,OAAO,CAAC,EAAE;QACV,MAAM,IAAI,KAAK,CAAC,0BAA0B,CAAC,CAAC;KAC7C;IACD,MAAM,EAAE,GAAG,IAAI,CAAC,aAAa,EAAoC,CAAC;IAClE,MAAM,MAAM,GAAG,aAAa,CAAC,EAAE,EAAE,MAAM,EAAE,OAAO,CAAC,CAAC;IAClD,MAAM,oBAAoB,GAAG,2BAA2B,CAAC,IAAI,EAAE,MAAM,CAAC,CAAC;IACvE,OAAO;QACL,GAAG,MAAM;QACT,eAAe,EAAE,oBAAoB;QACrC,UAAU,EAAE,oBAAoB,CAAC,MAAM,CAAC,CAAC,IAAI,EAAE,IAAI,EAAE,EAAE,CAAC,CAAC,IAAI,GAAG,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;KAC9D,CAAC;AAC9B,CAAC;AAnBD,kCAmBC;AAED;;;GAGG;AACH,SAAgB,SAAS,CACvB,MAA0C,EAC1C,IAAsB;IAEtB,MAAM,EAAE,KAAK,EAAE,GAAG,MAAM,CAAC;IACzB,IAAI,EAAE,CAAC;IACP,IAAI;QACF,EAAE,GAAG,IAAI,CAAC,wBAAwB,CAAC,KAAK,CAAC,CAAC;KAC3C;IAAC,OAAO,CAAC,EAAE;QACV,MAAM,IAAI,KAAK,CAAC,iCAAiC,CAAC,CAAC;KACpD;IACD,MAAM,MAAM,GAAG,aAAa,CAAC,EAAE,EAAE,MAAM,EAAE,IAAI,CAAC,OAAO,CAAC,CAAC;IACvD,MAAM,oBAAoB,GAAG,yBAAyB,CAAC,EAAE,EAAE,MAAM,EAAE,IAAI,CAAC,OAAO,CAAC,CAAC;IACjF,OAAO;QACL,GAAG,MAAM;QACT,eAAe,EAAE,oBAAoB;QACrC,UAAU,EAAE,oBAAoB,CAAC,MAAM,CAAC,CAAC,IAAI,EAAE,IAAI,EAAE,EAAE,CAAC,CAAC,IAAI,GAAG,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;KAC9D,CAAC;AAC9B,CAAC;AAlBD,8BAkBC","sourcesContent":["import * as utxolib from '@bitgo-beta/utxo-lib';\nimport { BitGoBase, IRequestTracer, Triple } from '@bitgo-beta/sdk-core';\nimport {\n  AbstractUtxoCoin,\n  ExplainTransactionOptions,\n  Output,\n  TransactionExplanation,\n  TransactionPrebuild,\n} from './abstractUtxoCoin';\nimport { bip32, BIP32Interface, bitgo } from '@bitgo-beta/utxo-lib';\n\n/**\n * Get the inputs for a psbt from a prebuild.\n */\nexport function getPsbtTxInputs(\n  psbtArg: string | utxolib.bitgo.UtxoPsbt,\n  network: utxolib.Network\n): { address: string; value: bigint; valueString: string }[] {\n  const psbt = psbtArg instanceof utxolib.bitgo.UtxoPsbt ? psbtArg : utxolib.bitgo.createPsbtFromHex(psbtArg, network);\n  const txInputs = psbt.txInputs;\n  return psbt.data.inputs.map((input, index) => {\n    let address: string;\n    let value: bigint;\n    if (input.witnessUtxo) {\n      address = utxolib.address.fromOutputScript(input.witnessUtxo.script, network);\n      value = input.witnessUtxo.value;\n    } else if (input.nonWitnessUtxo) {\n      const tx = utxolib.bitgo.createTransactionFromBuffer<bigint>(input.nonWitnessUtxo, network, {\n        amountType: 'bigint',\n      });\n      const txId = (Buffer.from(txInputs[index].hash).reverse() as Buffer).toString('hex');\n      if (tx.getId() !== txId) {\n        throw new Error('input transaction hex does not match id');\n      }\n      const prevTxOutputIndex = txInputs[index].index;\n      address = utxolib.address.fromOutputScript(tx.outs[prevTxOutputIndex].script, network);\n      value = tx.outs[prevTxOutputIndex].value;\n    } else {\n      throw new Error('psbt input is missing both witnessUtxo and nonWitnessUtxo');\n    }\n    return { address, value, valueString: value.toString() };\n  });\n}\n\n/**\n * Get the inputs for a transaction from a prebuild.\n */\nexport async function getTxInputs<TNumber extends number | bigint>(params: {\n  txPrebuild: TransactionPrebuild<TNumber>;\n  bitgo: BitGoBase;\n  coin: AbstractUtxoCoin;\n  disableNetworking: boolean;\n  reqId?: IRequestTracer;\n}): Promise<{ address: string; value: TNumber; valueString: string }[]> {\n  const { txPrebuild, bitgo, coin, disableNetworking, reqId } = params;\n  if (!txPrebuild.txHex) {\n    throw new Error(`txPrebuild.txHex not set`);\n  }\n  const transaction = coin.createTransactionFromHex<TNumber>(txPrebuild.txHex);\n  const transactionCache = {};\n  return await Promise.all(\n    transaction.ins.map(async (currentInput): Promise<{ address: string; value: TNumber; valueString: string }> => {\n      const transactionId = (Buffer.from(currentInput.hash).reverse() as Buffer).toString('hex');\n      const txHex = txPrebuild.txInfo?.txHexes?.[transactionId];\n      if (txHex) {\n        const localTx = coin.createTransactionFromHex<TNumber>(txHex);\n        if (localTx.getId() !== transactionId) {\n          throw new Error('input transaction hex does not match id');\n        }\n        const currentOutput = localTx.outs[currentInput.index];\n        const address = utxolib.address.fromOutputScript(currentOutput.script, coin.network);\n        return {\n          address,\n          value: currentOutput.value,\n          valueString: currentOutput.value.toString(),\n        };\n      } else if (!transactionCache[transactionId]) {\n        if (disableNetworking) {\n          throw new Error('attempting to retrieve transaction details externally with networking disabled');\n        }\n        if (reqId) {\n          bitgo.setRequestTracer(reqId);\n        }\n        transactionCache[transactionId] = await bitgo.get(coin.url(`/public/tx/${transactionId}`)).result();\n      }\n      const transactionDetails = transactionCache[transactionId];\n      return transactionDetails.outputs[currentInput.index];\n    })\n  );\n}\n\nfunction explainCommon<TNumber extends number | bigint>(\n  tx: bitgo.UtxoTransaction<TNumber>,\n  params: ExplainTransactionOptions<TNumber>,\n  network: utxolib.Network\n) {\n  const displayOrder = ['id', 'outputAmount', 'changeAmount', 'outputs', 'changeOutputs'];\n  let spendAmount = BigInt(0);\n  let changeAmount = BigInt(0);\n  const changeOutputs: Output[] = [];\n  const outputs: Output[] = [];\n\n  const { changeAddresses = [] } = params.txInfo ?? {};\n\n  tx.outs.forEach((currentOutput) => {\n    const currentAddress = utxolib.address.fromOutputScript(currentOutput.script, network);\n    const currentAmount = BigInt(currentOutput.value);\n\n    if (changeAddresses.includes(currentAddress)) {\n      // this is change\n      changeAmount += currentAmount;\n      changeOutputs.push({\n        address: currentAddress,\n        amount: currentAmount.toString(),\n      });\n      return;\n    }\n\n    spendAmount += currentAmount;\n    outputs.push({\n      address: currentAddress,\n      amount: currentAmount.toString(),\n    });\n  });\n\n  const outputDetails = {\n    outputAmount: spendAmount.toString(),\n    changeAmount: changeAmount.toString(),\n    outputs,\n    changeOutputs,\n  };\n\n  let fee: string | undefined;\n  let locktime: number | undefined;\n\n  if (params.feeInfo) {\n    displayOrder.push('fee');\n    fee = params.feeInfo;\n  }\n\n  if (Number.isInteger(tx.locktime) && tx.locktime > 0) {\n    displayOrder.push('locktime');\n    locktime = tx.locktime;\n  }\n\n  return { displayOrder, id: tx.getId(), ...outputDetails, fee, locktime };\n}\n\nfunction getRootWalletKeys<TNumber extends number | bigint>(params: ExplainTransactionOptions<TNumber>) {\n  const keys = params.pubs?.map((xpub) => bip32.fromBase58(xpub));\n  return keys && keys.length === 3 ? new bitgo.RootWalletKeys(keys as Triple<BIP32Interface>) : undefined;\n}\n\nfunction getPsbtInputSignaturesCount<TNumber extends number | bigint>(\n  psbt: bitgo.UtxoPsbt,\n  params: ExplainTransactionOptions<TNumber>\n) {\n  const rootWalletKeys = getRootWalletKeys(params);\n  return rootWalletKeys\n    ? bitgo.getSignatureValidationArrayPsbt(psbt, rootWalletKeys).map((sv) => sv[1].filter((v) => v).length)\n    : (Array(psbt.data.inputs.length) as number[]).fill(0);\n}\n\nfunction getTxInputSignaturesCount<TNumber extends number | bigint>(\n  tx: bitgo.UtxoTransaction<TNumber>,\n  params: ExplainTransactionOptions<TNumber>,\n  network: utxolib.Network\n) {\n  const prevOutputs = params.txInfo?.unspents?.map((u) => bitgo.toOutput<TNumber>(u, network));\n  const rootWalletKeys = getRootWalletKeys(params);\n  const { unspents = [] } = params.txInfo ?? {};\n\n  // get the number of signatures per input\n  return tx.ins.map((input, idx): number => {\n    if (unspents.length !== tx.ins.length) {\n      return 0;\n    }\n    if (!prevOutputs) {\n      throw new Error(`invalid state`);\n    }\n    if (!rootWalletKeys) {\n      // no pub keys or incorrect number of pub keys\n      return 0;\n    }\n    try {\n      return bitgo.verifySignatureWithUnspent<TNumber>(tx, idx, unspents, rootWalletKeys).filter((v) => v).length;\n    } catch (e) {\n      // some other error occurred and we can't validate the signatures\n      return 0;\n    }\n  });\n}\n\n/**\n * Decompose a raw psbt into useful information, such as the total amounts,\n * change amounts, and transaction outputs.\n */\nexport function explainPsbt<TNumber extends number | bigint>(\n  params: ExplainTransactionOptions<TNumber>,\n  network: utxolib.Network\n): TransactionExplanation {\n  const { txHex } = params;\n  let psbt: bitgo.UtxoPsbt;\n  try {\n    psbt = bitgo.createPsbtFromHex(txHex, network);\n  } catch (e) {\n    throw new Error('failed to parse psbt hex');\n  }\n  const tx = psbt.getUnsignedTx() as bitgo.UtxoTransaction<TNumber>;\n  const common = explainCommon(tx, params, network);\n  const inputSignaturesCount = getPsbtInputSignaturesCount(psbt, params);\n  return {\n    ...common,\n    inputSignatures: inputSignaturesCount,\n    signatures: inputSignaturesCount.reduce((prev, curr) => (curr > prev ? curr : prev), 0),\n  } as TransactionExplanation;\n}\n\n/**\n * Decompose a raw transaction into useful information, such as the total amounts,\n * change amounts, and transaction outputs.\n */\nexport function explainTx<TNumber extends number | bigint>(\n  params: ExplainTransactionOptions<TNumber>,\n  coin: AbstractUtxoCoin\n): TransactionExplanation {\n  const { txHex } = params;\n  let tx;\n  try {\n    tx = coin.createTransactionFromHex(txHex);\n  } catch (e) {\n    throw new Error('failed to parse transaction hex');\n  }\n  const common = explainCommon(tx, params, coin.network);\n  const inputSignaturesCount = getTxInputSignaturesCount(tx, params, coin.network);\n  return {\n    ...common,\n    inputSignatures: inputSignaturesCount,\n    signatures: inputSignaturesCount.reduce((prev, curr) => (curr > prev ? curr : prev), 0),\n  } as TransactionExplanation;\n}\n"]}
|