@bitgo-beta/abstract-utxo 1.6.1-alpha.425 → 1.6.1-alpha.427
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/dist/cjs/src/abstractUtxoCoin.d.ts +22 -5
- package/dist/cjs/src/abstractUtxoCoin.d.ts.map +1 -1
- package/dist/cjs/src/abstractUtxoCoin.js +53 -26
- package/dist/cjs/src/index.d.ts +2 -2
- package/dist/cjs/src/index.d.ts.map +1 -1
- package/dist/cjs/src/index.js +3 -3
- package/dist/cjs/src/recovery/backupKeyRecovery.js +4 -4
- package/dist/cjs/src/recovery/crossChainRecovery.d.ts +10 -0
- package/dist/cjs/src/recovery/crossChainRecovery.d.ts.map +1 -1
- package/dist/cjs/src/recovery/crossChainRecovery.js +48 -4
- package/dist/cjs/src/transaction/decode.d.ts +11 -0
- package/dist/cjs/src/transaction/decode.d.ts.map +1 -0
- package/dist/cjs/src/transaction/decode.js +81 -0
- package/dist/cjs/src/transaction/descriptor/verifyTransaction.d.ts.map +1 -1
- package/dist/cjs/src/transaction/descriptor/verifyTransaction.js +3 -2
- package/dist/cjs/src/transaction/explainTransaction.js +3 -3
- package/dist/cjs/src/transaction/fixedScript/SigningError.d.ts +24 -0
- package/dist/cjs/src/transaction/fixedScript/SigningError.d.ts.map +1 -0
- package/dist/cjs/src/transaction/fixedScript/SigningError.js +26 -0
- package/dist/cjs/src/transaction/fixedScript/explainPsbtWasm.d.ts +1 -1
- package/dist/cjs/src/transaction/fixedScript/explainPsbtWasm.d.ts.map +1 -1
- package/dist/cjs/src/transaction/fixedScript/explainPsbtWasm.js +1 -1
- package/dist/cjs/src/transaction/fixedScript/explainTransaction.d.ts.map +1 -1
- package/dist/cjs/src/transaction/fixedScript/explainTransaction.js +2 -2
- package/dist/cjs/src/transaction/fixedScript/index.d.ts +3 -0
- package/dist/cjs/src/transaction/fixedScript/index.d.ts.map +1 -1
- package/dist/cjs/src/transaction/fixedScript/index.js +18 -1
- package/dist/cjs/src/transaction/fixedScript/musig2.d.ts +4 -0
- package/dist/cjs/src/transaction/fixedScript/musig2.d.ts.map +1 -0
- package/dist/cjs/src/transaction/fixedScript/musig2.js +3 -0
- package/dist/cjs/src/transaction/fixedScript/replayProtection.d.ts +7 -0
- package/dist/cjs/src/transaction/fixedScript/replayProtection.d.ts.map +1 -0
- package/dist/cjs/src/transaction/fixedScript/replayProtection.js +78 -0
- package/dist/cjs/src/transaction/fixedScript/signLegacyTransaction.d.ts +30 -0
- package/dist/cjs/src/transaction/fixedScript/signLegacyTransaction.d.ts.map +1 -0
- package/dist/cjs/src/transaction/fixedScript/signLegacyTransaction.js +152 -0
- package/dist/cjs/src/transaction/fixedScript/signPsbt.d.ts +30 -0
- package/dist/cjs/src/transaction/fixedScript/signPsbt.d.ts.map +1 -0
- package/dist/cjs/src/transaction/fixedScript/signPsbt.js +174 -0
- package/dist/cjs/src/transaction/fixedScript/signPsbtWasm.d.ts +22 -0
- package/dist/cjs/src/transaction/fixedScript/signPsbtWasm.d.ts.map +1 -0
- package/dist/cjs/src/transaction/fixedScript/signPsbtWasm.js +129 -0
- package/dist/cjs/src/transaction/fixedScript/signTransaction.d.ts +5 -6
- package/dist/cjs/src/transaction/fixedScript/signTransaction.d.ts.map +1 -1
- package/dist/cjs/src/transaction/fixedScript/signTransaction.js +28 -79
- package/dist/cjs/src/transaction/fixedScript/verifyTransaction.d.ts.map +1 -1
- package/dist/cjs/src/transaction/fixedScript/verifyTransaction.js +3 -2
- package/dist/cjs/src/transaction/signTransaction.d.ts.map +1 -1
- package/dist/cjs/src/transaction/signTransaction.js +5 -2
- package/dist/cjs/src/transaction/types.d.ts +5 -0
- package/dist/cjs/src/transaction/types.d.ts.map +1 -1
- package/dist/cjs/src/transaction/types.js +5 -1
- package/dist/cjs/test/unit/recovery/crossChainRecovery.js +24 -1
- package/dist/cjs/test/unit/transaction/fixedScript/explainPsbt.js +3 -3
- package/dist/cjs/test/unit/transaction/fixedScript/parsePsbt.js +2 -2
- package/dist/cjs/test/unit/transaction/fixedScript/replayProtection.d.ts +2 -0
- package/dist/cjs/test/unit/transaction/fixedScript/replayProtection.d.ts.map +1 -0
- package/dist/cjs/test/unit/transaction/fixedScript/replayProtection.js +59 -0
- package/dist/cjs/test/unit/transaction/fixedScript/signPsbt.d.ts +2 -0
- package/dist/cjs/test/unit/transaction/fixedScript/signPsbt.d.ts.map +1 -0
- package/dist/cjs/test/unit/transaction/fixedScript/signPsbt.js +157 -0
- package/dist/cjs/test/unit/transaction/fixedScript/util.d.ts.map +1 -1
- package/dist/cjs/test/unit/transaction/fixedScript/util.js +2 -1
- package/dist/cjs/test/unit/transaction.js +65 -42
- package/dist/cjs/tsconfig.tsbuildinfo +1 -1
- package/dist/esm/abstractUtxoCoin.d.ts +22 -5
- package/dist/esm/abstractUtxoCoin.d.ts.map +1 -1
- package/dist/esm/abstractUtxoCoin.js +54 -27
- package/dist/esm/index.d.ts +2 -2
- package/dist/esm/index.d.ts.map +1 -1
- package/dist/esm/index.js +3 -3
- package/dist/esm/recovery/backupKeyRecovery.js +2 -2
- package/dist/esm/recovery/crossChainRecovery.d.ts +10 -0
- package/dist/esm/recovery/crossChainRecovery.d.ts.map +1 -1
- package/dist/esm/recovery/crossChainRecovery.js +46 -3
- package/dist/esm/transaction/decode.d.ts +11 -0
- package/dist/esm/transaction/decode.d.ts.map +1 -0
- package/dist/esm/transaction/decode.js +43 -0
- package/dist/esm/transaction/descriptor/verifyTransaction.d.ts.map +1 -1
- package/dist/esm/transaction/descriptor/verifyTransaction.js +3 -2
- package/dist/esm/transaction/explainTransaction.js +3 -3
- package/dist/esm/transaction/fixedScript/SigningError.d.ts +24 -0
- package/dist/esm/transaction/fixedScript/SigningError.d.ts.map +1 -0
- package/dist/esm/transaction/fixedScript/SigningError.js +21 -0
- package/dist/esm/transaction/fixedScript/explainPsbtWasm.d.ts +1 -1
- package/dist/esm/transaction/fixedScript/explainPsbtWasm.d.ts.map +1 -1
- package/dist/esm/transaction/fixedScript/explainPsbtWasm.js +1 -1
- package/dist/esm/transaction/fixedScript/explainTransaction.d.ts.map +1 -1
- package/dist/esm/transaction/fixedScript/explainTransaction.js +2 -2
- package/dist/esm/transaction/fixedScript/index.d.ts +3 -0
- package/dist/esm/transaction/fixedScript/index.d.ts.map +1 -1
- package/dist/esm/transaction/fixedScript/index.js +4 -1
- package/dist/esm/transaction/fixedScript/musig2.d.ts +4 -0
- package/dist/esm/transaction/fixedScript/musig2.d.ts.map +1 -0
- package/dist/esm/transaction/fixedScript/musig2.js +2 -0
- package/dist/esm/transaction/fixedScript/replayProtection.d.ts +7 -0
- package/dist/esm/transaction/fixedScript/replayProtection.d.ts.map +1 -0
- package/dist/esm/transaction/fixedScript/replayProtection.js +39 -0
- package/dist/esm/transaction/fixedScript/signLegacyTransaction.d.ts +30 -0
- package/dist/esm/transaction/fixedScript/signLegacyTransaction.d.ts.map +1 -0
- package/dist/esm/transaction/fixedScript/signLegacyTransaction.js +112 -0
- package/dist/esm/transaction/fixedScript/signPsbt.d.ts +30 -0
- package/dist/esm/transaction/fixedScript/signPsbt.d.ts.map +1 -0
- package/dist/esm/transaction/fixedScript/signPsbt.js +134 -0
- package/dist/esm/transaction/fixedScript/signPsbtWasm.d.ts +22 -0
- package/dist/esm/transaction/fixedScript/signPsbtWasm.d.ts.map +1 -0
- package/dist/esm/transaction/fixedScript/signPsbtWasm.js +122 -0
- package/dist/esm/transaction/fixedScript/signTransaction.d.ts +5 -6
- package/dist/esm/transaction/fixedScript/signTransaction.d.ts.map +1 -1
- package/dist/esm/transaction/fixedScript/signTransaction.js +28 -79
- package/dist/esm/transaction/fixedScript/verifyTransaction.d.ts.map +1 -1
- package/dist/esm/transaction/fixedScript/verifyTransaction.js +3 -2
- package/dist/esm/transaction/signTransaction.d.ts.map +1 -1
- package/dist/esm/transaction/signTransaction.js +5 -2
- package/dist/esm/transaction/types.d.ts +5 -0
- package/dist/esm/transaction/types.d.ts.map +1 -1
- package/dist/esm/transaction/types.js +4 -2
- package/package.json +13 -13
- package/dist/cjs/src/replayProtection.d.ts +0 -5
- package/dist/cjs/src/replayProtection.d.ts.map +0 -1
- package/dist/cjs/src/replayProtection.js +0 -58
- package/dist/cjs/src/sign.d.ts +0 -54
- package/dist/cjs/src/sign.d.ts.map +0 -1
- package/dist/cjs/src/sign.js +0 -205
- package/dist/esm/replayProtection.d.ts +0 -5
- package/dist/esm/replayProtection.d.ts.map +0 -1
- package/dist/esm/replayProtection.js +0 -20
- package/dist/esm/sign.d.ts +0 -54
- package/dist/esm/sign.d.ts.map +0 -1
- package/dist/esm/sign.js +0 -162
package/dist/cjs/src/sign.js
DELETED
|
@@ -1,205 +0,0 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
-
if (k2 === undefined) k2 = k;
|
|
4
|
-
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
-
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
-
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
-
}
|
|
8
|
-
Object.defineProperty(o, k2, desc);
|
|
9
|
-
}) : (function(o, m, k, k2) {
|
|
10
|
-
if (k2 === undefined) k2 = k;
|
|
11
|
-
o[k2] = m[k];
|
|
12
|
-
}));
|
|
13
|
-
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
14
|
-
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
15
|
-
}) : function(o, v) {
|
|
16
|
-
o["default"] = v;
|
|
17
|
-
});
|
|
18
|
-
var __importStar = (this && this.__importStar) || (function () {
|
|
19
|
-
var ownKeys = function(o) {
|
|
20
|
-
ownKeys = Object.getOwnPropertyNames || function (o) {
|
|
21
|
-
var ar = [];
|
|
22
|
-
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
|
|
23
|
-
return ar;
|
|
24
|
-
};
|
|
25
|
-
return ownKeys(o);
|
|
26
|
-
};
|
|
27
|
-
return function (mod) {
|
|
28
|
-
if (mod && mod.__esModule) return mod;
|
|
29
|
-
var result = {};
|
|
30
|
-
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
|
|
31
|
-
__setModuleDefault(result, mod);
|
|
32
|
-
return result;
|
|
33
|
-
};
|
|
34
|
-
})();
|
|
35
|
-
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
36
|
-
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
37
|
-
};
|
|
38
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
|
39
|
-
exports.TransactionSigningError = exports.InputSigningError = void 0;
|
|
40
|
-
exports.signAndVerifyPsbt = signAndVerifyPsbt;
|
|
41
|
-
exports.signAndVerifyWalletTransaction = signAndVerifyWalletTransaction;
|
|
42
|
-
const utxolib = __importStar(require("@bitgo-beta/utxo-lib"));
|
|
43
|
-
const debug_1 = __importDefault(require("debug"));
|
|
44
|
-
const replayProtection_1 = require("./replayProtection");
|
|
45
|
-
const debug = (0, debug_1.default)('bitgo:v2:utxo');
|
|
46
|
-
const { isWalletUnspent, signInputWithUnspent, toOutput } = utxolib.bitgo;
|
|
47
|
-
class InputSigningError extends Error {
|
|
48
|
-
static expectedWalletUnspent(inputIndex, unspent) {
|
|
49
|
-
return new InputSigningError(inputIndex, unspent, `not a wallet unspent, not a replay protection unspent`);
|
|
50
|
-
}
|
|
51
|
-
constructor(inputIndex, unspent, reason) {
|
|
52
|
-
super(`signing error at input ${inputIndex}: unspentId=${unspent.id}: ${reason}`);
|
|
53
|
-
this.inputIndex = inputIndex;
|
|
54
|
-
this.unspent = unspent;
|
|
55
|
-
this.reason = reason;
|
|
56
|
-
}
|
|
57
|
-
}
|
|
58
|
-
exports.InputSigningError = InputSigningError;
|
|
59
|
-
class TransactionSigningError extends Error {
|
|
60
|
-
constructor(signErrors, verifyError) {
|
|
61
|
-
super(`sign errors at inputs: [${signErrors.join(',')}], ` +
|
|
62
|
-
`verify errors at inputs: [${verifyError.join(',')}], see log for details`);
|
|
63
|
-
}
|
|
64
|
-
}
|
|
65
|
-
exports.TransactionSigningError = TransactionSigningError;
|
|
66
|
-
/**
|
|
67
|
-
* Sign all inputs of a psbt and verify signatures after signing.
|
|
68
|
-
* Collects and logs signing errors and verification errors, throws error in the end if any of them
|
|
69
|
-
* failed.
|
|
70
|
-
*
|
|
71
|
-
* If it is the last signature, finalize and extract the transaction from the psbt.
|
|
72
|
-
*
|
|
73
|
-
* This function mirrors signAndVerifyWalletTransaction, but is used for signing PSBTs instead of
|
|
74
|
-
* using TransactionBuilder
|
|
75
|
-
*
|
|
76
|
-
* @param psbt
|
|
77
|
-
* @param signerKeychain
|
|
78
|
-
* @param isLastSignature
|
|
79
|
-
*/
|
|
80
|
-
function signAndVerifyPsbt(psbt, signerKeychain, { isLastSignature,
|
|
81
|
-
/** deprecated */
|
|
82
|
-
allowNonSegwitSigningWithoutPrevTx, }) {
|
|
83
|
-
const txInputs = psbt.txInputs;
|
|
84
|
-
const outputIds = [];
|
|
85
|
-
const scriptTypes = [];
|
|
86
|
-
const signErrors = psbt.data.inputs
|
|
87
|
-
.map((input, inputIndex) => {
|
|
88
|
-
const outputId = utxolib.bitgo.formatOutputId(utxolib.bitgo.getOutputIdForInput(txInputs[inputIndex]));
|
|
89
|
-
outputIds.push(outputId);
|
|
90
|
-
const { scriptType } = utxolib.bitgo.parsePsbtInput(input);
|
|
91
|
-
scriptTypes.push(scriptType);
|
|
92
|
-
if (scriptType === 'p2shP2pk') {
|
|
93
|
-
debug('Skipping signature for input %d of %d (RP input?)', inputIndex + 1, psbt.data.inputs.length);
|
|
94
|
-
return;
|
|
95
|
-
}
|
|
96
|
-
try {
|
|
97
|
-
psbt.signInputHD(inputIndex, signerKeychain);
|
|
98
|
-
debug('Successfully signed input %d of %d', inputIndex + 1, psbt.data.inputs.length);
|
|
99
|
-
}
|
|
100
|
-
catch (e) {
|
|
101
|
-
return new InputSigningError(inputIndex, { id: outputId }, e);
|
|
102
|
-
}
|
|
103
|
-
})
|
|
104
|
-
.filter((e) => e !== undefined);
|
|
105
|
-
const verifyErrors = psbt.data.inputs
|
|
106
|
-
.map((input, inputIndex) => {
|
|
107
|
-
const scriptType = scriptTypes[inputIndex];
|
|
108
|
-
if (scriptType === 'p2shP2pk') {
|
|
109
|
-
debug('Skipping input signature %d of %d (unspent from replay protection address which is platform signed only)', inputIndex + 1, psbt.data.inputs.length);
|
|
110
|
-
return;
|
|
111
|
-
}
|
|
112
|
-
const outputId = outputIds[inputIndex];
|
|
113
|
-
try {
|
|
114
|
-
if (!psbt.validateSignaturesOfInputHD(inputIndex, signerKeychain)) {
|
|
115
|
-
return new InputSigningError(inputIndex, { id: outputId }, new Error(`invalid signature`));
|
|
116
|
-
}
|
|
117
|
-
}
|
|
118
|
-
catch (e) {
|
|
119
|
-
debug('Invalid signature');
|
|
120
|
-
return new InputSigningError(inputIndex, { id: outputId }, e);
|
|
121
|
-
}
|
|
122
|
-
})
|
|
123
|
-
.filter((e) => e !== undefined);
|
|
124
|
-
if (signErrors.length || verifyErrors.length) {
|
|
125
|
-
throw new TransactionSigningError(signErrors, verifyErrors);
|
|
126
|
-
}
|
|
127
|
-
if (isLastSignature) {
|
|
128
|
-
psbt.finalizeAllInputs();
|
|
129
|
-
return psbt.extractTransaction();
|
|
130
|
-
}
|
|
131
|
-
return psbt;
|
|
132
|
-
}
|
|
133
|
-
/**
|
|
134
|
-
* Sign all inputs of a wallet transaction and verify signatures after signing.
|
|
135
|
-
* Collects and logs signing errors and verification errors, throws error in the end if any of them
|
|
136
|
-
* failed.
|
|
137
|
-
*
|
|
138
|
-
* @param transaction - wallet transaction (builder) to be signed
|
|
139
|
-
* @param unspents - transaction unspents
|
|
140
|
-
* @param walletSigner - signing parameters
|
|
141
|
-
* @param isLastSignature - Returns full-signed transaction when true. Builds half-signed when false.
|
|
142
|
-
*/
|
|
143
|
-
function signAndVerifyWalletTransaction(transaction, unspents, walletSigner, { isLastSignature }) {
|
|
144
|
-
const network = transaction.network;
|
|
145
|
-
const prevOutputs = unspents.map((u) => toOutput(u, network));
|
|
146
|
-
let txBuilder;
|
|
147
|
-
if (transaction instanceof utxolib.bitgo.UtxoTransaction) {
|
|
148
|
-
txBuilder = utxolib.bitgo.createTransactionBuilderFromTransaction(transaction, prevOutputs);
|
|
149
|
-
if (transaction.ins.length !== unspents.length) {
|
|
150
|
-
throw new Error(`transaction inputs must match unspents`);
|
|
151
|
-
}
|
|
152
|
-
}
|
|
153
|
-
else if (transaction instanceof utxolib.bitgo.UtxoTransactionBuilder) {
|
|
154
|
-
txBuilder = transaction;
|
|
155
|
-
}
|
|
156
|
-
else {
|
|
157
|
-
throw new Error(`must pass UtxoTransaction or UtxoTransactionBuilder`);
|
|
158
|
-
}
|
|
159
|
-
const signErrors = unspents
|
|
160
|
-
.map((unspent, inputIndex) => {
|
|
161
|
-
if ((0, replayProtection_1.isReplayProtectionUnspent)(unspent, network)) {
|
|
162
|
-
debug('Skipping signature for input %d of %d (RP input?)', inputIndex + 1, unspents.length);
|
|
163
|
-
return;
|
|
164
|
-
}
|
|
165
|
-
if (!isWalletUnspent(unspent)) {
|
|
166
|
-
return InputSigningError.expectedWalletUnspent(inputIndex, unspent);
|
|
167
|
-
}
|
|
168
|
-
try {
|
|
169
|
-
signInputWithUnspent(txBuilder, inputIndex, unspent, walletSigner);
|
|
170
|
-
debug('Successfully signed input %d of %d', inputIndex + 1, unspents.length);
|
|
171
|
-
}
|
|
172
|
-
catch (e) {
|
|
173
|
-
return new InputSigningError(inputIndex, unspent, e);
|
|
174
|
-
}
|
|
175
|
-
})
|
|
176
|
-
.filter((e) => e !== undefined);
|
|
177
|
-
const signedTransaction = isLastSignature ? txBuilder.build() : txBuilder.buildIncomplete();
|
|
178
|
-
const verifyErrors = signedTransaction.ins
|
|
179
|
-
.map((input, inputIndex) => {
|
|
180
|
-
const unspent = unspents[inputIndex];
|
|
181
|
-
if ((0, replayProtection_1.isReplayProtectionUnspent)(unspent, network)) {
|
|
182
|
-
debug('Skipping input signature %d of %d (unspent from replay protection address which is platform signed only)', inputIndex + 1, unspents.length);
|
|
183
|
-
return;
|
|
184
|
-
}
|
|
185
|
-
if (!isWalletUnspent(unspent)) {
|
|
186
|
-
return InputSigningError.expectedWalletUnspent(inputIndex, unspent);
|
|
187
|
-
}
|
|
188
|
-
try {
|
|
189
|
-
const publicKey = walletSigner.deriveForChainAndIndex(unspent.chain, unspent.index).signer.publicKey;
|
|
190
|
-
if (!utxolib.bitgo.verifySignatureWithPublicKey(signedTransaction, inputIndex, prevOutputs, publicKey)) {
|
|
191
|
-
return new InputSigningError(inputIndex, unspent, new Error(`invalid signature`));
|
|
192
|
-
}
|
|
193
|
-
}
|
|
194
|
-
catch (e) {
|
|
195
|
-
debug('Invalid signature');
|
|
196
|
-
return new InputSigningError(inputIndex, unspent, e);
|
|
197
|
-
}
|
|
198
|
-
})
|
|
199
|
-
.filter((e) => e !== undefined);
|
|
200
|
-
if (signErrors.length || verifyErrors.length) {
|
|
201
|
-
throw new TransactionSigningError(signErrors, verifyErrors);
|
|
202
|
-
}
|
|
203
|
-
return signedTransaction;
|
|
204
|
-
}
|
|
205
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sign.js","sourceRoot":"","sources":["../../../src/sign.ts"],"names":[],"mappings":"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","sourcesContent":["import * as utxolib from '@bitgo-beta/utxo-lib';\nimport debugLib from 'debug';\n\nimport { isReplayProtectionUnspent } from './replayProtection';\n\nconst debug = debugLib('bitgo:v2:utxo');\n\nconst { isWalletUnspent, signInputWithUnspent, toOutput } = utxolib.bitgo;\n\ntype Unspent<TNumber extends number | bigint = number> = utxolib.bitgo.Unspent<TNumber>;\n\ntype RootWalletKeys = utxolib.bitgo.RootWalletKeys;\n\ntype PsbtParsedScriptTypes =\n  | 'p2sh'\n  | 'p2wsh'\n  | 'p2shP2wsh'\n  | 'p2shP2pk'\n  | 'taprootKeyPathSpend'\n  | 'taprootScriptPathSpend';\n\nexport class InputSigningError<TNumber extends number | bigint = number> extends Error {\n  static expectedWalletUnspent<TNumber extends number | bigint>(\n    inputIndex: number,\n    unspent: Unspent<TNumber> | { id: string }\n  ): InputSigningError<TNumber> {\n    return new InputSigningError(inputIndex, unspent, `not a wallet unspent, not a replay protection unspent`);\n  }\n\n  constructor(\n    public inputIndex: number,\n    public unspent: Unspent<TNumber> | { id: string },\n    public reason: Error | string\n  ) {\n    super(`signing error at input ${inputIndex}: unspentId=${unspent.id}: ${reason}`);\n  }\n}\n\nexport class TransactionSigningError<TNumber extends number | bigint = number> extends Error {\n  constructor(signErrors: InputSigningError<TNumber>[], verifyError: InputSigningError<TNumber>[]) {\n    super(\n      `sign errors at inputs: [${signErrors.join(',')}], ` +\n        `verify errors at inputs: [${verifyError.join(',')}], see log for details`\n    );\n  }\n}\n\n/**\n * Sign all inputs of a psbt and verify signatures after signing.\n * Collects and logs signing errors and verification errors, throws error in the end if any of them\n * failed.\n *\n * If it is the last signature, finalize and extract the transaction from the psbt.\n *\n * This function mirrors signAndVerifyWalletTransaction, but is used for signing PSBTs instead of\n * using TransactionBuilder\n *\n * @param psbt\n * @param signerKeychain\n * @param isLastSignature\n */\nexport function signAndVerifyPsbt(\n  psbt: utxolib.bitgo.UtxoPsbt,\n  signerKeychain: utxolib.BIP32Interface,\n  {\n    isLastSignature,\n    /** deprecated */\n    allowNonSegwitSigningWithoutPrevTx,\n  }: { isLastSignature: boolean; allowNonSegwitSigningWithoutPrevTx?: boolean }\n): utxolib.bitgo.UtxoPsbt | utxolib.bitgo.UtxoTransaction<bigint> {\n  const txInputs = psbt.txInputs;\n  const outputIds: string[] = [];\n  const scriptTypes: PsbtParsedScriptTypes[] = [];\n\n  const signErrors: InputSigningError<bigint>[] = psbt.data.inputs\n    .map((input, inputIndex: number) => {\n      const outputId = utxolib.bitgo.formatOutputId(utxolib.bitgo.getOutputIdForInput(txInputs[inputIndex]));\n      outputIds.push(outputId);\n\n      const { scriptType } = utxolib.bitgo.parsePsbtInput(input);\n      scriptTypes.push(scriptType);\n\n      if (scriptType === 'p2shP2pk') {\n        debug('Skipping signature for input %d of %d (RP input?)', inputIndex + 1, psbt.data.inputs.length);\n        return;\n      }\n\n      try {\n        psbt.signInputHD(inputIndex, signerKeychain);\n        debug('Successfully signed input %d of %d', inputIndex + 1, psbt.data.inputs.length);\n      } catch (e) {\n        return new InputSigningError<bigint>(inputIndex, { id: outputId }, e);\n      }\n    })\n    .filter((e): e is InputSigningError<bigint> => e !== undefined);\n\n  const verifyErrors: InputSigningError<bigint>[] = psbt.data.inputs\n    .map((input, inputIndex) => {\n      const scriptType = scriptTypes[inputIndex];\n      if (scriptType === 'p2shP2pk') {\n        debug(\n          'Skipping input signature %d of %d (unspent from replay protection address which is platform signed only)',\n          inputIndex + 1,\n          psbt.data.inputs.length\n        );\n        return;\n      }\n\n      const outputId = outputIds[inputIndex];\n      try {\n        if (!psbt.validateSignaturesOfInputHD(inputIndex, signerKeychain)) {\n          return new InputSigningError(inputIndex, { id: outputId }, new Error(`invalid signature`));\n        }\n      } catch (e) {\n        debug('Invalid signature');\n        return new InputSigningError<bigint>(inputIndex, { id: outputId }, e);\n      }\n    })\n    .filter((e): e is InputSigningError<bigint> => e !== undefined);\n\n  if (signErrors.length || verifyErrors.length) {\n    throw new TransactionSigningError(signErrors, verifyErrors);\n  }\n\n  if (isLastSignature) {\n    psbt.finalizeAllInputs();\n    return psbt.extractTransaction();\n  }\n\n  return psbt;\n}\n\n/**\n * Sign all inputs of a wallet transaction and verify signatures after signing.\n * Collects and logs signing errors and verification errors, throws error in the end if any of them\n * failed.\n *\n * @param transaction - wallet transaction (builder) to be signed\n * @param unspents - transaction unspents\n * @param walletSigner - signing parameters\n * @param isLastSignature - Returns full-signed transaction when true. Builds half-signed when false.\n */\nexport function signAndVerifyWalletTransaction<TNumber extends number | bigint>(\n  transaction: utxolib.bitgo.UtxoTransaction<TNumber> | utxolib.bitgo.UtxoTransactionBuilder<TNumber>,\n  unspents: Unspent<TNumber>[],\n  walletSigner: utxolib.bitgo.WalletUnspentSigner<RootWalletKeys>,\n  { isLastSignature }: { isLastSignature: boolean }\n): utxolib.bitgo.UtxoTransaction<TNumber> {\n  const network = transaction.network as utxolib.Network;\n  const prevOutputs = unspents.map((u) => toOutput(u, network));\n\n  let txBuilder: utxolib.bitgo.UtxoTransactionBuilder<TNumber>;\n  if (transaction instanceof utxolib.bitgo.UtxoTransaction) {\n    txBuilder = utxolib.bitgo.createTransactionBuilderFromTransaction<TNumber>(transaction, prevOutputs);\n    if (transaction.ins.length !== unspents.length) {\n      throw new Error(`transaction inputs must match unspents`);\n    }\n  } else if (transaction instanceof utxolib.bitgo.UtxoTransactionBuilder) {\n    txBuilder = transaction;\n  } else {\n    throw new Error(`must pass UtxoTransaction or UtxoTransactionBuilder`);\n  }\n\n  const signErrors: InputSigningError<TNumber>[] = unspents\n    .map((unspent: Unspent<TNumber>, inputIndex: number) => {\n      if (isReplayProtectionUnspent<TNumber>(unspent, network)) {\n        debug('Skipping signature for input %d of %d (RP input?)', inputIndex + 1, unspents.length);\n        return;\n      }\n      if (!isWalletUnspent<TNumber>(unspent)) {\n        return InputSigningError.expectedWalletUnspent<TNumber>(inputIndex, unspent);\n      }\n      try {\n        signInputWithUnspent<TNumber>(txBuilder, inputIndex, unspent, walletSigner);\n        debug('Successfully signed input %d of %d', inputIndex + 1, unspents.length);\n      } catch (e) {\n        return new InputSigningError<TNumber>(inputIndex, unspent, e);\n      }\n    })\n    .filter((e): e is InputSigningError<TNumber> => e !== undefined);\n\n  const signedTransaction = isLastSignature ? txBuilder.build() : txBuilder.buildIncomplete();\n\n  const verifyErrors: InputSigningError<TNumber>[] = signedTransaction.ins\n    .map((input, inputIndex) => {\n      const unspent = unspents[inputIndex] as Unspent<TNumber>;\n      if (isReplayProtectionUnspent<TNumber>(unspent, network)) {\n        debug(\n          'Skipping input signature %d of %d (unspent from replay protection address which is platform signed only)',\n          inputIndex + 1,\n          unspents.length\n        );\n        return;\n      }\n      if (!isWalletUnspent<TNumber>(unspent)) {\n        return InputSigningError.expectedWalletUnspent<TNumber>(inputIndex, unspent);\n      }\n      try {\n        const publicKey = walletSigner.deriveForChainAndIndex(unspent.chain, unspent.index).signer.publicKey;\n        if (\n          !utxolib.bitgo.verifySignatureWithPublicKey<TNumber>(signedTransaction, inputIndex, prevOutputs, publicKey)\n        ) {\n          return new InputSigningError(inputIndex, unspent, new Error(`invalid signature`));\n        }\n      } catch (e) {\n        debug('Invalid signature');\n        return new InputSigningError<TNumber>(inputIndex, unspent, e);\n      }\n    })\n    .filter((e): e is InputSigningError<TNumber> => e !== undefined);\n\n  if (signErrors.length || verifyErrors.length) {\n    throw new TransactionSigningError(signErrors, verifyErrors);\n  }\n\n  return signedTransaction;\n}\n"]}
|
|
@@ -1,5 +0,0 @@
|
|
|
1
|
-
import * as utxolib from '@bitgo-beta/utxo-lib';
|
|
2
|
-
export declare function getReplayProtectionAddresses(network: utxolib.Network): string[];
|
|
3
|
-
export declare function getReplayProtectionOutputScripts(network: utxolib.Network): Buffer[];
|
|
4
|
-
export declare function isReplayProtectionUnspent<TNumber extends number | bigint>(u: utxolib.bitgo.Unspent<TNumber>, network: utxolib.Network): boolean;
|
|
5
|
-
//# sourceMappingURL=replayProtection.d.ts.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"replayProtection.d.ts","sourceRoot":"","sources":["../../src/replayProtection.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,OAAO,MAAM,sBAAsB,CAAC;AAEhD,wBAAgB,4BAA4B,CAAC,OAAO,EAAE,OAAO,CAAC,OAAO,GAAG,MAAM,EAAE,CAW/E;AAED,wBAAgB,gCAAgC,CAAC,OAAO,EAAE,OAAO,CAAC,OAAO,GAAG,MAAM,EAAE,CAInF;AAED,wBAAgB,yBAAyB,CAAC,OAAO,SAAS,MAAM,GAAG,MAAM,EACvE,CAAC,EAAE,OAAO,CAAC,KAAK,CAAC,OAAO,CAAC,OAAO,CAAC,EACjC,OAAO,EAAE,OAAO,CAAC,OAAO,GACvB,OAAO,CAET"}
|
|
@@ -1,20 +0,0 @@
|
|
|
1
|
-
import * as wasmUtxo from '@bitgo/wasm-utxo';
|
|
2
|
-
import * as utxolib from '@bitgo-beta/utxo-lib';
|
|
3
|
-
export function getReplayProtectionAddresses(network) {
|
|
4
|
-
switch (network) {
|
|
5
|
-
case utxolib.networks.bitcoincash:
|
|
6
|
-
case utxolib.networks.bitcoinsv:
|
|
7
|
-
return ['33p1q7mTGyeM5UnZERGiMcVUkY12SCsatA'];
|
|
8
|
-
case utxolib.networks.bitcoincashTestnet:
|
|
9
|
-
case utxolib.networks.bitcoinsvTestnet:
|
|
10
|
-
return ['2MuMnPoSDgWEpNWH28X2nLtYMXQJCyT61eY'];
|
|
11
|
-
}
|
|
12
|
-
return [];
|
|
13
|
-
}
|
|
14
|
-
export function getReplayProtectionOutputScripts(network) {
|
|
15
|
-
return getReplayProtectionAddresses(network).map((address) => Buffer.from(wasmUtxo.utxolibCompat.toOutputScript(address, network)));
|
|
16
|
-
}
|
|
17
|
-
export function isReplayProtectionUnspent(u, network) {
|
|
18
|
-
return getReplayProtectionAddresses(network).includes(u.address);
|
|
19
|
-
}
|
|
20
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmVwbGF5UHJvdGVjdGlvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9yZXBsYXlQcm90ZWN0aW9uLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sS0FBSyxRQUFRLE1BQU0sa0JBQWtCLENBQUM7QUFDN0MsT0FBTyxLQUFLLE9BQU8sTUFBTSxzQkFBc0IsQ0FBQztBQUVoRCxNQUFNLFVBQVUsNEJBQTRCLENBQUMsT0FBd0I7SUFDbkUsUUFBUSxPQUFPLEVBQUUsQ0FBQztRQUNoQixLQUFLLE9BQU8sQ0FBQyxRQUFRLENBQUMsV0FBVyxDQUFDO1FBQ2xDLEtBQUssT0FBTyxDQUFDLFFBQVEsQ0FBQyxTQUFTO1lBQzdCLE9BQU8sQ0FBQyxvQ0FBb0MsQ0FBQyxDQUFDO1FBQ2hELEtBQUssT0FBTyxDQUFDLFFBQVEsQ0FBQyxrQkFBa0IsQ0FBQztRQUN6QyxLQUFLLE9BQU8sQ0FBQyxRQUFRLENBQUMsZ0JBQWdCO1lBQ3BDLE9BQU8sQ0FBQyxxQ0FBcUMsQ0FBQyxDQUFDO0lBQ25ELENBQUM7SUFFRCxPQUFPLEVBQUUsQ0FBQztBQUNaLENBQUM7QUFFRCxNQUFNLFVBQVUsZ0NBQWdDLENBQUMsT0FBd0I7SUFDdkUsT0FBTyw0QkFBNEIsQ0FBQyxPQUFPLENBQUMsQ0FBQyxHQUFHLENBQUMsQ0FBQyxPQUFPLEVBQUUsRUFBRSxDQUMzRCxNQUFNLENBQUMsSUFBSSxDQUFDLFFBQVEsQ0FBQyxhQUFhLENBQUMsY0FBYyxDQUFDLE9BQU8sRUFBRSxPQUFPLENBQUMsQ0FBQyxDQUNyRSxDQUFDO0FBQ0osQ0FBQztBQUVELE1BQU0sVUFBVSx5QkFBeUIsQ0FDdkMsQ0FBaUMsRUFDakMsT0FBd0I7SUFFeEIsT0FBTyw0QkFBNEIsQ0FBQyxPQUFPLENBQUMsQ0FBQyxRQUFRLENBQUMsQ0FBQyxDQUFDLE9BQU8sQ0FBQyxDQUFDO0FBQ25FLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgKiBhcyB3YXNtVXR4byBmcm9tICdAYml0Z28vd2FzbS11dHhvJztcbmltcG9ydCAqIGFzIHV0eG9saWIgZnJvbSAnQGJpdGdvLWJldGEvdXR4by1saWInO1xuXG5leHBvcnQgZnVuY3Rpb24gZ2V0UmVwbGF5UHJvdGVjdGlvbkFkZHJlc3NlcyhuZXR3b3JrOiB1dHhvbGliLk5ldHdvcmspOiBzdHJpbmdbXSB7XG4gIHN3aXRjaCAobmV0d29yaykge1xuICAgIGNhc2UgdXR4b2xpYi5uZXR3b3Jrcy5iaXRjb2luY2FzaDpcbiAgICBjYXNlIHV0eG9saWIubmV0d29ya3MuYml0Y29pbnN2OlxuICAgICAgcmV0dXJuIFsnMzNwMXE3bVRHeWVNNVVuWkVSR2lNY1ZVa1kxMlNDc2F0QSddO1xuICAgIGNhc2UgdXR4b2xpYi5uZXR3b3Jrcy5iaXRjb2luY2FzaFRlc3RuZXQ6XG4gICAgY2FzZSB1dHhvbGliLm5ldHdvcmtzLmJpdGNvaW5zdlRlc3RuZXQ6XG4gICAgICByZXR1cm4gWycyTXVNblBvU0RnV0VwTldIMjhYMm5MdFlNWFFKQ3lUNjFlWSddO1xuICB9XG5cbiAgcmV0dXJuIFtdO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gZ2V0UmVwbGF5UHJvdGVjdGlvbk91dHB1dFNjcmlwdHMobmV0d29yazogdXR4b2xpYi5OZXR3b3JrKTogQnVmZmVyW10ge1xuICByZXR1cm4gZ2V0UmVwbGF5UHJvdGVjdGlvbkFkZHJlc3NlcyhuZXR3b3JrKS5tYXAoKGFkZHJlc3MpID0+XG4gICAgQnVmZmVyLmZyb20od2FzbVV0eG8udXR4b2xpYkNvbXBhdC50b091dHB1dFNjcmlwdChhZGRyZXNzLCBuZXR3b3JrKSlcbiAgKTtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGlzUmVwbGF5UHJvdGVjdGlvblVuc3BlbnQ8VE51bWJlciBleHRlbmRzIG51bWJlciB8IGJpZ2ludD4oXG4gIHU6IHV0eG9saWIuYml0Z28uVW5zcGVudDxUTnVtYmVyPixcbiAgbmV0d29yazogdXR4b2xpYi5OZXR3b3JrXG4pOiBib29sZWFuIHtcbiAgcmV0dXJuIGdldFJlcGxheVByb3RlY3Rpb25BZGRyZXNzZXMobmV0d29yaykuaW5jbHVkZXModS5hZGRyZXNzKTtcbn1cbiJdfQ==
|
package/dist/esm/sign.d.ts
DELETED
|
@@ -1,54 +0,0 @@
|
|
|
1
|
-
import * as utxolib from '@bitgo-beta/utxo-lib';
|
|
2
|
-
type Unspent<TNumber extends number | bigint = number> = utxolib.bitgo.Unspent<TNumber>;
|
|
3
|
-
type RootWalletKeys = utxolib.bitgo.RootWalletKeys;
|
|
4
|
-
export declare class InputSigningError<TNumber extends number | bigint = number> extends Error {
|
|
5
|
-
inputIndex: number;
|
|
6
|
-
unspent: Unspent<TNumber> | {
|
|
7
|
-
id: string;
|
|
8
|
-
};
|
|
9
|
-
reason: Error | string;
|
|
10
|
-
static expectedWalletUnspent<TNumber extends number | bigint>(inputIndex: number, unspent: Unspent<TNumber> | {
|
|
11
|
-
id: string;
|
|
12
|
-
}): InputSigningError<TNumber>;
|
|
13
|
-
constructor(inputIndex: number, unspent: Unspent<TNumber> | {
|
|
14
|
-
id: string;
|
|
15
|
-
}, reason: Error | string);
|
|
16
|
-
}
|
|
17
|
-
export declare class TransactionSigningError<TNumber extends number | bigint = number> extends Error {
|
|
18
|
-
constructor(signErrors: InputSigningError<TNumber>[], verifyError: InputSigningError<TNumber>[]);
|
|
19
|
-
}
|
|
20
|
-
/**
|
|
21
|
-
* Sign all inputs of a psbt and verify signatures after signing.
|
|
22
|
-
* Collects and logs signing errors and verification errors, throws error in the end if any of them
|
|
23
|
-
* failed.
|
|
24
|
-
*
|
|
25
|
-
* If it is the last signature, finalize and extract the transaction from the psbt.
|
|
26
|
-
*
|
|
27
|
-
* This function mirrors signAndVerifyWalletTransaction, but is used for signing PSBTs instead of
|
|
28
|
-
* using TransactionBuilder
|
|
29
|
-
*
|
|
30
|
-
* @param psbt
|
|
31
|
-
* @param signerKeychain
|
|
32
|
-
* @param isLastSignature
|
|
33
|
-
*/
|
|
34
|
-
export declare function signAndVerifyPsbt(psbt: utxolib.bitgo.UtxoPsbt, signerKeychain: utxolib.BIP32Interface, { isLastSignature,
|
|
35
|
-
/** deprecated */
|
|
36
|
-
allowNonSegwitSigningWithoutPrevTx, }: {
|
|
37
|
-
isLastSignature: boolean;
|
|
38
|
-
allowNonSegwitSigningWithoutPrevTx?: boolean;
|
|
39
|
-
}): utxolib.bitgo.UtxoPsbt | utxolib.bitgo.UtxoTransaction<bigint>;
|
|
40
|
-
/**
|
|
41
|
-
* Sign all inputs of a wallet transaction and verify signatures after signing.
|
|
42
|
-
* Collects and logs signing errors and verification errors, throws error in the end if any of them
|
|
43
|
-
* failed.
|
|
44
|
-
*
|
|
45
|
-
* @param transaction - wallet transaction (builder) to be signed
|
|
46
|
-
* @param unspents - transaction unspents
|
|
47
|
-
* @param walletSigner - signing parameters
|
|
48
|
-
* @param isLastSignature - Returns full-signed transaction when true. Builds half-signed when false.
|
|
49
|
-
*/
|
|
50
|
-
export declare function signAndVerifyWalletTransaction<TNumber extends number | bigint>(transaction: utxolib.bitgo.UtxoTransaction<TNumber> | utxolib.bitgo.UtxoTransactionBuilder<TNumber>, unspents: Unspent<TNumber>[], walletSigner: utxolib.bitgo.WalletUnspentSigner<RootWalletKeys>, { isLastSignature }: {
|
|
51
|
-
isLastSignature: boolean;
|
|
52
|
-
}): utxolib.bitgo.UtxoTransaction<TNumber>;
|
|
53
|
-
export {};
|
|
54
|
-
//# sourceMappingURL=sign.d.ts.map
|
package/dist/esm/sign.d.ts.map
DELETED
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"sign.d.ts","sourceRoot":"","sources":["../../src/sign.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,OAAO,MAAM,sBAAsB,CAAC;AAShD,KAAK,OAAO,CAAC,OAAO,SAAS,MAAM,GAAG,MAAM,GAAG,MAAM,IAAI,OAAO,CAAC,KAAK,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;AAExF,KAAK,cAAc,GAAG,OAAO,CAAC,KAAK,CAAC,cAAc,CAAC;AAUnD,qBAAa,iBAAiB,CAAC,OAAO,SAAS,MAAM,GAAG,MAAM,GAAG,MAAM,CAAE,SAAQ,KAAK;IAS3E,UAAU,EAAE,MAAM;IAClB,OAAO,EAAE,OAAO,CAAC,OAAO,CAAC,GAAG;QAAE,EAAE,EAAE,MAAM,CAAA;KAAE;IAC1C,MAAM,EAAE,KAAK,GAAG,MAAM;IAV/B,MAAM,CAAC,qBAAqB,CAAC,OAAO,SAAS,MAAM,GAAG,MAAM,EAC1D,UAAU,EAAE,MAAM,EAClB,OAAO,EAAE,OAAO,CAAC,OAAO,CAAC,GAAG;QAAE,EAAE,EAAE,MAAM,CAAA;KAAE,GACzC,iBAAiB,CAAC,OAAO,CAAC;gBAKpB,UAAU,EAAE,MAAM,EAClB,OAAO,EAAE,OAAO,CAAC,OAAO,CAAC,GAAG;QAAE,EAAE,EAAE,MAAM,CAAA;KAAE,EAC1C,MAAM,EAAE,KAAK,GAAG,MAAM;CAIhC;AAED,qBAAa,uBAAuB,CAAC,OAAO,SAAS,MAAM,GAAG,MAAM,GAAG,MAAM,CAAE,SAAQ,KAAK;gBAC9E,UAAU,EAAE,iBAAiB,CAAC,OAAO,CAAC,EAAE,EAAE,WAAW,EAAE,iBAAiB,CAAC,OAAO,CAAC,EAAE;CAMhG;AAED;;;;;;;;;;;;;GAaG;AACH,wBAAgB,iBAAiB,CAC/B,IAAI,EAAE,OAAO,CAAC,KAAK,CAAC,QAAQ,EAC5B,cAAc,EAAE,OAAO,CAAC,cAAc,EACtC,EACE,eAAe;AACf,iBAAiB;AACjB,kCAAkC,GACnC,EAAE;IAAE,eAAe,EAAE,OAAO,CAAC;IAAC,kCAAkC,CAAC,EAAE,OAAO,CAAA;CAAE,GAC5E,OAAO,CAAC,KAAK,CAAC,QAAQ,GAAG,OAAO,CAAC,KAAK,CAAC,eAAe,CAAC,MAAM,CAAC,CA6DhE;AAED;;;;;;;;;GASG;AACH,wBAAgB,8BAA8B,CAAC,OAAO,SAAS,MAAM,GAAG,MAAM,EAC5E,WAAW,EAAE,OAAO,CAAC,KAAK,CAAC,eAAe,CAAC,OAAO,CAAC,GAAG,OAAO,CAAC,KAAK,CAAC,sBAAsB,CAAC,OAAO,CAAC,EACnG,QAAQ,EAAE,OAAO,CAAC,OAAO,CAAC,EAAE,EAC5B,YAAY,EAAE,OAAO,CAAC,KAAK,CAAC,mBAAmB,CAAC,cAAc,CAAC,EAC/D,EAAE,eAAe,EAAE,EAAE;IAAE,eAAe,EAAE,OAAO,CAAA;CAAE,GAChD,OAAO,CAAC,KAAK,CAAC,eAAe,CAAC,OAAO,CAAC,CAqExC"}
|
package/dist/esm/sign.js
DELETED
|
@@ -1,162 +0,0 @@
|
|
|
1
|
-
import * as utxolib from '@bitgo-beta/utxo-lib';
|
|
2
|
-
import debugLib from 'debug';
|
|
3
|
-
import { isReplayProtectionUnspent } from './replayProtection';
|
|
4
|
-
const debug = debugLib('bitgo:v2:utxo');
|
|
5
|
-
const { isWalletUnspent, signInputWithUnspent, toOutput } = utxolib.bitgo;
|
|
6
|
-
export class InputSigningError extends Error {
|
|
7
|
-
static expectedWalletUnspent(inputIndex, unspent) {
|
|
8
|
-
return new InputSigningError(inputIndex, unspent, `not a wallet unspent, not a replay protection unspent`);
|
|
9
|
-
}
|
|
10
|
-
constructor(inputIndex, unspent, reason) {
|
|
11
|
-
super(`signing error at input ${inputIndex}: unspentId=${unspent.id}: ${reason}`);
|
|
12
|
-
this.inputIndex = inputIndex;
|
|
13
|
-
this.unspent = unspent;
|
|
14
|
-
this.reason = reason;
|
|
15
|
-
}
|
|
16
|
-
}
|
|
17
|
-
export class TransactionSigningError extends Error {
|
|
18
|
-
constructor(signErrors, verifyError) {
|
|
19
|
-
super(`sign errors at inputs: [${signErrors.join(',')}], ` +
|
|
20
|
-
`verify errors at inputs: [${verifyError.join(',')}], see log for details`);
|
|
21
|
-
}
|
|
22
|
-
}
|
|
23
|
-
/**
|
|
24
|
-
* Sign all inputs of a psbt and verify signatures after signing.
|
|
25
|
-
* Collects and logs signing errors and verification errors, throws error in the end if any of them
|
|
26
|
-
* failed.
|
|
27
|
-
*
|
|
28
|
-
* If it is the last signature, finalize and extract the transaction from the psbt.
|
|
29
|
-
*
|
|
30
|
-
* This function mirrors signAndVerifyWalletTransaction, but is used for signing PSBTs instead of
|
|
31
|
-
* using TransactionBuilder
|
|
32
|
-
*
|
|
33
|
-
* @param psbt
|
|
34
|
-
* @param signerKeychain
|
|
35
|
-
* @param isLastSignature
|
|
36
|
-
*/
|
|
37
|
-
export function signAndVerifyPsbt(psbt, signerKeychain, { isLastSignature,
|
|
38
|
-
/** deprecated */
|
|
39
|
-
allowNonSegwitSigningWithoutPrevTx, }) {
|
|
40
|
-
const txInputs = psbt.txInputs;
|
|
41
|
-
const outputIds = [];
|
|
42
|
-
const scriptTypes = [];
|
|
43
|
-
const signErrors = psbt.data.inputs
|
|
44
|
-
.map((input, inputIndex) => {
|
|
45
|
-
const outputId = utxolib.bitgo.formatOutputId(utxolib.bitgo.getOutputIdForInput(txInputs[inputIndex]));
|
|
46
|
-
outputIds.push(outputId);
|
|
47
|
-
const { scriptType } = utxolib.bitgo.parsePsbtInput(input);
|
|
48
|
-
scriptTypes.push(scriptType);
|
|
49
|
-
if (scriptType === 'p2shP2pk') {
|
|
50
|
-
debug('Skipping signature for input %d of %d (RP input?)', inputIndex + 1, psbt.data.inputs.length);
|
|
51
|
-
return;
|
|
52
|
-
}
|
|
53
|
-
try {
|
|
54
|
-
psbt.signInputHD(inputIndex, signerKeychain);
|
|
55
|
-
debug('Successfully signed input %d of %d', inputIndex + 1, psbt.data.inputs.length);
|
|
56
|
-
}
|
|
57
|
-
catch (e) {
|
|
58
|
-
return new InputSigningError(inputIndex, { id: outputId }, e);
|
|
59
|
-
}
|
|
60
|
-
})
|
|
61
|
-
.filter((e) => e !== undefined);
|
|
62
|
-
const verifyErrors = psbt.data.inputs
|
|
63
|
-
.map((input, inputIndex) => {
|
|
64
|
-
const scriptType = scriptTypes[inputIndex];
|
|
65
|
-
if (scriptType === 'p2shP2pk') {
|
|
66
|
-
debug('Skipping input signature %d of %d (unspent from replay protection address which is platform signed only)', inputIndex + 1, psbt.data.inputs.length);
|
|
67
|
-
return;
|
|
68
|
-
}
|
|
69
|
-
const outputId = outputIds[inputIndex];
|
|
70
|
-
try {
|
|
71
|
-
if (!psbt.validateSignaturesOfInputHD(inputIndex, signerKeychain)) {
|
|
72
|
-
return new InputSigningError(inputIndex, { id: outputId }, new Error(`invalid signature`));
|
|
73
|
-
}
|
|
74
|
-
}
|
|
75
|
-
catch (e) {
|
|
76
|
-
debug('Invalid signature');
|
|
77
|
-
return new InputSigningError(inputIndex, { id: outputId }, e);
|
|
78
|
-
}
|
|
79
|
-
})
|
|
80
|
-
.filter((e) => e !== undefined);
|
|
81
|
-
if (signErrors.length || verifyErrors.length) {
|
|
82
|
-
throw new TransactionSigningError(signErrors, verifyErrors);
|
|
83
|
-
}
|
|
84
|
-
if (isLastSignature) {
|
|
85
|
-
psbt.finalizeAllInputs();
|
|
86
|
-
return psbt.extractTransaction();
|
|
87
|
-
}
|
|
88
|
-
return psbt;
|
|
89
|
-
}
|
|
90
|
-
/**
|
|
91
|
-
* Sign all inputs of a wallet transaction and verify signatures after signing.
|
|
92
|
-
* Collects and logs signing errors and verification errors, throws error in the end if any of them
|
|
93
|
-
* failed.
|
|
94
|
-
*
|
|
95
|
-
* @param transaction - wallet transaction (builder) to be signed
|
|
96
|
-
* @param unspents - transaction unspents
|
|
97
|
-
* @param walletSigner - signing parameters
|
|
98
|
-
* @param isLastSignature - Returns full-signed transaction when true. Builds half-signed when false.
|
|
99
|
-
*/
|
|
100
|
-
export function signAndVerifyWalletTransaction(transaction, unspents, walletSigner, { isLastSignature }) {
|
|
101
|
-
const network = transaction.network;
|
|
102
|
-
const prevOutputs = unspents.map((u) => toOutput(u, network));
|
|
103
|
-
let txBuilder;
|
|
104
|
-
if (transaction instanceof utxolib.bitgo.UtxoTransaction) {
|
|
105
|
-
txBuilder = utxolib.bitgo.createTransactionBuilderFromTransaction(transaction, prevOutputs);
|
|
106
|
-
if (transaction.ins.length !== unspents.length) {
|
|
107
|
-
throw new Error(`transaction inputs must match unspents`);
|
|
108
|
-
}
|
|
109
|
-
}
|
|
110
|
-
else if (transaction instanceof utxolib.bitgo.UtxoTransactionBuilder) {
|
|
111
|
-
txBuilder = transaction;
|
|
112
|
-
}
|
|
113
|
-
else {
|
|
114
|
-
throw new Error(`must pass UtxoTransaction or UtxoTransactionBuilder`);
|
|
115
|
-
}
|
|
116
|
-
const signErrors = unspents
|
|
117
|
-
.map((unspent, inputIndex) => {
|
|
118
|
-
if (isReplayProtectionUnspent(unspent, network)) {
|
|
119
|
-
debug('Skipping signature for input %d of %d (RP input?)', inputIndex + 1, unspents.length);
|
|
120
|
-
return;
|
|
121
|
-
}
|
|
122
|
-
if (!isWalletUnspent(unspent)) {
|
|
123
|
-
return InputSigningError.expectedWalletUnspent(inputIndex, unspent);
|
|
124
|
-
}
|
|
125
|
-
try {
|
|
126
|
-
signInputWithUnspent(txBuilder, inputIndex, unspent, walletSigner);
|
|
127
|
-
debug('Successfully signed input %d of %d', inputIndex + 1, unspents.length);
|
|
128
|
-
}
|
|
129
|
-
catch (e) {
|
|
130
|
-
return new InputSigningError(inputIndex, unspent, e);
|
|
131
|
-
}
|
|
132
|
-
})
|
|
133
|
-
.filter((e) => e !== undefined);
|
|
134
|
-
const signedTransaction = isLastSignature ? txBuilder.build() : txBuilder.buildIncomplete();
|
|
135
|
-
const verifyErrors = signedTransaction.ins
|
|
136
|
-
.map((input, inputIndex) => {
|
|
137
|
-
const unspent = unspents[inputIndex];
|
|
138
|
-
if (isReplayProtectionUnspent(unspent, network)) {
|
|
139
|
-
debug('Skipping input signature %d of %d (unspent from replay protection address which is platform signed only)', inputIndex + 1, unspents.length);
|
|
140
|
-
return;
|
|
141
|
-
}
|
|
142
|
-
if (!isWalletUnspent(unspent)) {
|
|
143
|
-
return InputSigningError.expectedWalletUnspent(inputIndex, unspent);
|
|
144
|
-
}
|
|
145
|
-
try {
|
|
146
|
-
const publicKey = walletSigner.deriveForChainAndIndex(unspent.chain, unspent.index).signer.publicKey;
|
|
147
|
-
if (!utxolib.bitgo.verifySignatureWithPublicKey(signedTransaction, inputIndex, prevOutputs, publicKey)) {
|
|
148
|
-
return new InputSigningError(inputIndex, unspent, new Error(`invalid signature`));
|
|
149
|
-
}
|
|
150
|
-
}
|
|
151
|
-
catch (e) {
|
|
152
|
-
debug('Invalid signature');
|
|
153
|
-
return new InputSigningError(inputIndex, unspent, e);
|
|
154
|
-
}
|
|
155
|
-
})
|
|
156
|
-
.filter((e) => e !== undefined);
|
|
157
|
-
if (signErrors.length || verifyErrors.length) {
|
|
158
|
-
throw new TransactionSigningError(signErrors, verifyErrors);
|
|
159
|
-
}
|
|
160
|
-
return signedTransaction;
|
|
161
|
-
}
|
|
162
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sign.js","sourceRoot":"","sources":["../../src/sign.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,OAAO,MAAM,sBAAsB,CAAC;AAChD,OAAO,QAAQ,MAAM,OAAO,CAAC;AAE7B,OAAO,EAAE,yBAAyB,EAAE,MAAM,oBAAoB,CAAC;AAE/D,MAAM,KAAK,GAAG,QAAQ,CAAC,eAAe,CAAC,CAAC;AAExC,MAAM,EAAE,eAAe,EAAE,oBAAoB,EAAE,QAAQ,EAAE,GAAG,OAAO,CAAC,KAAK,CAAC;AAc1E,MAAM,OAAO,iBAA4D,SAAQ,KAAK;IACpF,MAAM,CAAC,qBAAqB,CAC1B,UAAkB,EAClB,OAA0C;QAE1C,OAAO,IAAI,iBAAiB,CAAC,UAAU,EAAE,OAAO,EAAE,uDAAuD,CAAC,CAAC;IAC7G,CAAC;IAED,YACS,UAAkB,EAClB,OAA0C,EAC1C,MAAsB;QAE7B,KAAK,CAAC,0BAA0B,UAAU,eAAe,OAAO,CAAC,EAAE,KAAK,MAAM,EAAE,CAAC,CAAC;QAJ3E,eAAU,GAAV,UAAU,CAAQ;QAClB,YAAO,GAAP,OAAO,CAAmC;QAC1C,WAAM,GAAN,MAAM,CAAgB;IAG/B,CAAC;CACF;AAED,MAAM,OAAO,uBAAkE,SAAQ,KAAK;IAC1F,YAAY,UAAwC,EAAE,WAAyC;QAC7F,KAAK,CACH,2BAA2B,UAAU,CAAC,IAAI,CAAC,GAAG,CAAC,KAAK;YAClD,6BAA6B,WAAW,CAAC,IAAI,CAAC,GAAG,CAAC,wBAAwB,CAC7E,CAAC;IACJ,CAAC;CACF;AAED;;;;;;;;;;;;;GAaG;AACH,MAAM,UAAU,iBAAiB,CAC/B,IAA4B,EAC5B,cAAsC,EACtC,EACE,eAAe;AACf,iBAAiB;AACjB,kCAAkC,GACyC;IAE7E,MAAM,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC;IAC/B,MAAM,SAAS,GAAa,EAAE,CAAC;IAC/B,MAAM,WAAW,GAA4B,EAAE,CAAC;IAEhD,MAAM,UAAU,GAAgC,IAAI,CAAC,IAAI,CAAC,MAAM;SAC7D,GAAG,CAAC,CAAC,KAAK,EAAE,UAAkB,EAAE,EAAE;QACjC,MAAM,QAAQ,GAAG,OAAO,CAAC,KAAK,CAAC,cAAc,CAAC,OAAO,CAAC,KAAK,CAAC,mBAAmB,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC;QACvG,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;QAEzB,MAAM,EAAE,UAAU,EAAE,GAAG,OAAO,CAAC,KAAK,CAAC,cAAc,CAAC,KAAK,CAAC,CAAC;QAC3D,WAAW,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;QAE7B,IAAI,UAAU,KAAK,UAAU,EAAE,CAAC;YAC9B,KAAK,CAAC,mDAAmD,EAAE,UAAU,GAAG,CAAC,EAAE,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;YACpG,OAAO;QACT,CAAC;QAED,IAAI,CAAC;YACH,IAAI,CAAC,WAAW,CAAC,UAAU,EAAE,cAAc,CAAC,CAAC;YAC7C,KAAK,CAAC,oCAAoC,EAAE,UAAU,GAAG,CAAC,EAAE,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;QACvF,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,OAAO,IAAI,iBAAiB,CAAS,UAAU,EAAE,EAAE,EAAE,EAAE,QAAQ,EAAE,EAAE,CAAC,CAAC,CAAC;QACxE,CAAC;IACH,CAAC,CAAC;SACD,MAAM,CAAC,CAAC,CAAC,EAAkC,EAAE,CAAC,CAAC,KAAK,SAAS,CAAC,CAAC;IAElE,MAAM,YAAY,GAAgC,IAAI,CAAC,IAAI,CAAC,MAAM;SAC/D,GAAG,CAAC,CAAC,KAAK,EAAE,UAAU,EAAE,EAAE;QACzB,MAAM,UAAU,GAAG,WAAW,CAAC,UAAU,CAAC,CAAC;QAC3C,IAAI,UAAU,KAAK,UAAU,EAAE,CAAC;YAC9B,KAAK,CACH,0GAA0G,EAC1G,UAAU,GAAG,CAAC,EACd,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,MAAM,CACxB,CAAC;YACF,OAAO;QACT,CAAC;QAED,MAAM,QAAQ,GAAG,SAAS,CAAC,UAAU,CAAC,CAAC;QACvC,IAAI,CAAC;YACH,IAAI,CAAC,IAAI,CAAC,2BAA2B,CAAC,UAAU,EAAE,cAAc,CAAC,EAAE,CAAC;gBAClE,OAAO,IAAI,iBAAiB,CAAC,UAAU,EAAE,EAAE,EAAE,EAAE,QAAQ,EAAE,EAAE,IAAI,KAAK,CAAC,mBAAmB,CAAC,CAAC,CAAC;YAC7F,CAAC;QACH,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,KAAK,CAAC,mBAAmB,CAAC,CAAC;YAC3B,OAAO,IAAI,iBAAiB,CAAS,UAAU,EAAE,EAAE,EAAE,EAAE,QAAQ,EAAE,EAAE,CAAC,CAAC,CAAC;QACxE,CAAC;IACH,CAAC,CAAC;SACD,MAAM,CAAC,CAAC,CAAC,EAAkC,EAAE,CAAC,CAAC,KAAK,SAAS,CAAC,CAAC;IAElE,IAAI,UAAU,CAAC,MAAM,IAAI,YAAY,CAAC,MAAM,EAAE,CAAC;QAC7C,MAAM,IAAI,uBAAuB,CAAC,UAAU,EAAE,YAAY,CAAC,CAAC;IAC9D,CAAC;IAED,IAAI,eAAe,EAAE,CAAC;QACpB,IAAI,CAAC,iBAAiB,EAAE,CAAC;QACzB,OAAO,IAAI,CAAC,kBAAkB,EAAE,CAAC;IACnC,CAAC;IAED,OAAO,IAAI,CAAC;AACd,CAAC;AAED;;;;;;;;;GASG;AACH,MAAM,UAAU,8BAA8B,CAC5C,WAAmG,EACnG,QAA4B,EAC5B,YAA+D,EAC/D,EAAE,eAAe,EAAgC;IAEjD,MAAM,OAAO,GAAG,WAAW,CAAC,OAA0B,CAAC;IACvD,MAAM,WAAW,GAAG,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,QAAQ,CAAC,CAAC,EAAE,OAAO,CAAC,CAAC,CAAC;IAE9D,IAAI,SAAwD,CAAC;IAC7D,IAAI,WAAW,YAAY,OAAO,CAAC,KAAK,CAAC,eAAe,EAAE,CAAC;QACzD,SAAS,GAAG,OAAO,CAAC,KAAK,CAAC,uCAAuC,CAAU,WAAW,EAAE,WAAW,CAAC,CAAC;QACrG,IAAI,WAAW,CAAC,GAAG,CAAC,MAAM,KAAK,QAAQ,CAAC,MAAM,EAAE,CAAC;YAC/C,MAAM,IAAI,KAAK,CAAC,wCAAwC,CAAC,CAAC;QAC5D,CAAC;IACH,CAAC;SAAM,IAAI,WAAW,YAAY,OAAO,CAAC,KAAK,CAAC,sBAAsB,EAAE,CAAC;QACvE,SAAS,GAAG,WAAW,CAAC;IAC1B,CAAC;SAAM,CAAC;QACN,MAAM,IAAI,KAAK,CAAC,qDAAqD,CAAC,CAAC;IACzE,CAAC;IAED,MAAM,UAAU,GAAiC,QAAQ;SACtD,GAAG,CAAC,CAAC,OAAyB,EAAE,UAAkB,EAAE,EAAE;QACrD,IAAI,yBAAyB,CAAU,OAAO,EAAE,OAAO,CAAC,EAAE,CAAC;YACzD,KAAK,CAAC,mDAAmD,EAAE,UAAU,GAAG,CAAC,EAAE,QAAQ,CAAC,MAAM,CAAC,CAAC;YAC5F,OAAO;QACT,CAAC;QACD,IAAI,CAAC,eAAe,CAAU,OAAO,CAAC,EAAE,CAAC;YACvC,OAAO,iBAAiB,CAAC,qBAAqB,CAAU,UAAU,EAAE,OAAO,CAAC,CAAC;QAC/E,CAAC;QACD,IAAI,CAAC;YACH,oBAAoB,CAAU,SAAS,EAAE,UAAU,EAAE,OAAO,EAAE,YAAY,CAAC,CAAC;YAC5E,KAAK,CAAC,oCAAoC,EAAE,UAAU,GAAG,CAAC,EAAE,QAAQ,CAAC,MAAM,CAAC,CAAC;QAC/E,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,OAAO,IAAI,iBAAiB,CAAU,UAAU,EAAE,OAAO,EAAE,CAAC,CAAC,CAAC;QAChE,CAAC;IACH,CAAC,CAAC;SACD,MAAM,CAAC,CAAC,CAAC,EAAmC,EAAE,CAAC,CAAC,KAAK,SAAS,CAAC,CAAC;IAEnE,MAAM,iBAAiB,GAAG,eAAe,CAAC,CAAC,CAAC,SAAS,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC,eAAe,EAAE,CAAC;IAE5F,MAAM,YAAY,GAAiC,iBAAiB,CAAC,GAAG;SACrE,GAAG,CAAC,CAAC,KAAK,EAAE,UAAU,EAAE,EAAE;QACzB,MAAM,OAAO,GAAG,QAAQ,CAAC,UAAU,CAAqB,CAAC;QACzD,IAAI,yBAAyB,CAAU,OAAO,EAAE,OAAO,CAAC,EAAE,CAAC;YACzD,KAAK,CACH,0GAA0G,EAC1G,UAAU,GAAG,CAAC,EACd,QAAQ,CAAC,MAAM,CAChB,CAAC;YACF,OAAO;QACT,CAAC;QACD,IAAI,CAAC,eAAe,CAAU,OAAO,CAAC,EAAE,CAAC;YACvC,OAAO,iBAAiB,CAAC,qBAAqB,CAAU,UAAU,EAAE,OAAO,CAAC,CAAC;QAC/E,CAAC;QACD,IAAI,CAAC;YACH,MAAM,SAAS,GAAG,YAAY,CAAC,sBAAsB,CAAC,OAAO,CAAC,KAAK,EAAE,OAAO,CAAC,KAAK,CAAC,CAAC,MAAM,CAAC,SAAS,CAAC;YACrG,IACE,CAAC,OAAO,CAAC,KAAK,CAAC,4BAA4B,CAAU,iBAAiB,EAAE,UAAU,EAAE,WAAW,EAAE,SAAS,CAAC,EAC3G,CAAC;gBACD,OAAO,IAAI,iBAAiB,CAAC,UAAU,EAAE,OAAO,EAAE,IAAI,KAAK,CAAC,mBAAmB,CAAC,CAAC,CAAC;YACpF,CAAC;QACH,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,KAAK,CAAC,mBAAmB,CAAC,CAAC;YAC3B,OAAO,IAAI,iBAAiB,CAAU,UAAU,EAAE,OAAO,EAAE,CAAC,CAAC,CAAC;QAChE,CAAC;IACH,CAAC,CAAC;SACD,MAAM,CAAC,CAAC,CAAC,EAAmC,EAAE,CAAC,CAAC,KAAK,SAAS,CAAC,CAAC;IAEnE,IAAI,UAAU,CAAC,MAAM,IAAI,YAAY,CAAC,MAAM,EAAE,CAAC;QAC7C,MAAM,IAAI,uBAAuB,CAAC,UAAU,EAAE,YAAY,CAAC,CAAC;IAC9D,CAAC;IAED,OAAO,iBAAiB,CAAC;AAC3B,CAAC","sourcesContent":["import * as utxolib from '@bitgo-beta/utxo-lib';\nimport debugLib from 'debug';\n\nimport { isReplayProtectionUnspent } from './replayProtection';\n\nconst debug = debugLib('bitgo:v2:utxo');\n\nconst { isWalletUnspent, signInputWithUnspent, toOutput } = utxolib.bitgo;\n\ntype Unspent<TNumber extends number | bigint = number> = utxolib.bitgo.Unspent<TNumber>;\n\ntype RootWalletKeys = utxolib.bitgo.RootWalletKeys;\n\ntype PsbtParsedScriptTypes =\n  | 'p2sh'\n  | 'p2wsh'\n  | 'p2shP2wsh'\n  | 'p2shP2pk'\n  | 'taprootKeyPathSpend'\n  | 'taprootScriptPathSpend';\n\nexport class InputSigningError<TNumber extends number | bigint = number> extends Error {\n  static expectedWalletUnspent<TNumber extends number | bigint>(\n    inputIndex: number,\n    unspent: Unspent<TNumber> | { id: string }\n  ): InputSigningError<TNumber> {\n    return new InputSigningError(inputIndex, unspent, `not a wallet unspent, not a replay protection unspent`);\n  }\n\n  constructor(\n    public inputIndex: number,\n    public unspent: Unspent<TNumber> | { id: string },\n    public reason: Error | string\n  ) {\n    super(`signing error at input ${inputIndex}: unspentId=${unspent.id}: ${reason}`);\n  }\n}\n\nexport class TransactionSigningError<TNumber extends number | bigint = number> extends Error {\n  constructor(signErrors: InputSigningError<TNumber>[], verifyError: InputSigningError<TNumber>[]) {\n    super(\n      `sign errors at inputs: [${signErrors.join(',')}], ` +\n        `verify errors at inputs: [${verifyError.join(',')}], see log for details`\n    );\n  }\n}\n\n/**\n * Sign all inputs of a psbt and verify signatures after signing.\n * Collects and logs signing errors and verification errors, throws error in the end if any of them\n * failed.\n *\n * If it is the last signature, finalize and extract the transaction from the psbt.\n *\n * This function mirrors signAndVerifyWalletTransaction, but is used for signing PSBTs instead of\n * using TransactionBuilder\n *\n * @param psbt\n * @param signerKeychain\n * @param isLastSignature\n */\nexport function signAndVerifyPsbt(\n  psbt: utxolib.bitgo.UtxoPsbt,\n  signerKeychain: utxolib.BIP32Interface,\n  {\n    isLastSignature,\n    /** deprecated */\n    allowNonSegwitSigningWithoutPrevTx,\n  }: { isLastSignature: boolean; allowNonSegwitSigningWithoutPrevTx?: boolean }\n): utxolib.bitgo.UtxoPsbt | utxolib.bitgo.UtxoTransaction<bigint> {\n  const txInputs = psbt.txInputs;\n  const outputIds: string[] = [];\n  const scriptTypes: PsbtParsedScriptTypes[] = [];\n\n  const signErrors: InputSigningError<bigint>[] = psbt.data.inputs\n    .map((input, inputIndex: number) => {\n      const outputId = utxolib.bitgo.formatOutputId(utxolib.bitgo.getOutputIdForInput(txInputs[inputIndex]));\n      outputIds.push(outputId);\n\n      const { scriptType } = utxolib.bitgo.parsePsbtInput(input);\n      scriptTypes.push(scriptType);\n\n      if (scriptType === 'p2shP2pk') {\n        debug('Skipping signature for input %d of %d (RP input?)', inputIndex + 1, psbt.data.inputs.length);\n        return;\n      }\n\n      try {\n        psbt.signInputHD(inputIndex, signerKeychain);\n        debug('Successfully signed input %d of %d', inputIndex + 1, psbt.data.inputs.length);\n      } catch (e) {\n        return new InputSigningError<bigint>(inputIndex, { id: outputId }, e);\n      }\n    })\n    .filter((e): e is InputSigningError<bigint> => e !== undefined);\n\n  const verifyErrors: InputSigningError<bigint>[] = psbt.data.inputs\n    .map((input, inputIndex) => {\n      const scriptType = scriptTypes[inputIndex];\n      if (scriptType === 'p2shP2pk') {\n        debug(\n          'Skipping input signature %d of %d (unspent from replay protection address which is platform signed only)',\n          inputIndex + 1,\n          psbt.data.inputs.length\n        );\n        return;\n      }\n\n      const outputId = outputIds[inputIndex];\n      try {\n        if (!psbt.validateSignaturesOfInputHD(inputIndex, signerKeychain)) {\n          return new InputSigningError(inputIndex, { id: outputId }, new Error(`invalid signature`));\n        }\n      } catch (e) {\n        debug('Invalid signature');\n        return new InputSigningError<bigint>(inputIndex, { id: outputId }, e);\n      }\n    })\n    .filter((e): e is InputSigningError<bigint> => e !== undefined);\n\n  if (signErrors.length || verifyErrors.length) {\n    throw new TransactionSigningError(signErrors, verifyErrors);\n  }\n\n  if (isLastSignature) {\n    psbt.finalizeAllInputs();\n    return psbt.extractTransaction();\n  }\n\n  return psbt;\n}\n\n/**\n * Sign all inputs of a wallet transaction and verify signatures after signing.\n * Collects and logs signing errors and verification errors, throws error in the end if any of them\n * failed.\n *\n * @param transaction - wallet transaction (builder) to be signed\n * @param unspents - transaction unspents\n * @param walletSigner - signing parameters\n * @param isLastSignature - Returns full-signed transaction when true. Builds half-signed when false.\n */\nexport function signAndVerifyWalletTransaction<TNumber extends number | bigint>(\n  transaction: utxolib.bitgo.UtxoTransaction<TNumber> | utxolib.bitgo.UtxoTransactionBuilder<TNumber>,\n  unspents: Unspent<TNumber>[],\n  walletSigner: utxolib.bitgo.WalletUnspentSigner<RootWalletKeys>,\n  { isLastSignature }: { isLastSignature: boolean }\n): utxolib.bitgo.UtxoTransaction<TNumber> {\n  const network = transaction.network as utxolib.Network;\n  const prevOutputs = unspents.map((u) => toOutput(u, network));\n\n  let txBuilder: utxolib.bitgo.UtxoTransactionBuilder<TNumber>;\n  if (transaction instanceof utxolib.bitgo.UtxoTransaction) {\n    txBuilder = utxolib.bitgo.createTransactionBuilderFromTransaction<TNumber>(transaction, prevOutputs);\n    if (transaction.ins.length !== unspents.length) {\n      throw new Error(`transaction inputs must match unspents`);\n    }\n  } else if (transaction instanceof utxolib.bitgo.UtxoTransactionBuilder) {\n    txBuilder = transaction;\n  } else {\n    throw new Error(`must pass UtxoTransaction or UtxoTransactionBuilder`);\n  }\n\n  const signErrors: InputSigningError<TNumber>[] = unspents\n    .map((unspent: Unspent<TNumber>, inputIndex: number) => {\n      if (isReplayProtectionUnspent<TNumber>(unspent, network)) {\n        debug('Skipping signature for input %d of %d (RP input?)', inputIndex + 1, unspents.length);\n        return;\n      }\n      if (!isWalletUnspent<TNumber>(unspent)) {\n        return InputSigningError.expectedWalletUnspent<TNumber>(inputIndex, unspent);\n      }\n      try {\n        signInputWithUnspent<TNumber>(txBuilder, inputIndex, unspent, walletSigner);\n        debug('Successfully signed input %d of %d', inputIndex + 1, unspents.length);\n      } catch (e) {\n        return new InputSigningError<TNumber>(inputIndex, unspent, e);\n      }\n    })\n    .filter((e): e is InputSigningError<TNumber> => e !== undefined);\n\n  const signedTransaction = isLastSignature ? txBuilder.build() : txBuilder.buildIncomplete();\n\n  const verifyErrors: InputSigningError<TNumber>[] = signedTransaction.ins\n    .map((input, inputIndex) => {\n      const unspent = unspents[inputIndex] as Unspent<TNumber>;\n      if (isReplayProtectionUnspent<TNumber>(unspent, network)) {\n        debug(\n          'Skipping input signature %d of %d (unspent from replay protection address which is platform signed only)',\n          inputIndex + 1,\n          unspents.length\n        );\n        return;\n      }\n      if (!isWalletUnspent<TNumber>(unspent)) {\n        return InputSigningError.expectedWalletUnspent<TNumber>(inputIndex, unspent);\n      }\n      try {\n        const publicKey = walletSigner.deriveForChainAndIndex(unspent.chain, unspent.index).signer.publicKey;\n        if (\n          !utxolib.bitgo.verifySignatureWithPublicKey<TNumber>(signedTransaction, inputIndex, prevOutputs, publicKey)\n        ) {\n          return new InputSigningError(inputIndex, unspent, new Error(`invalid signature`));\n        }\n      } catch (e) {\n        debug('Invalid signature');\n        return new InputSigningError<TNumber>(inputIndex, unspent, e);\n      }\n    })\n    .filter((e): e is InputSigningError<TNumber> => e !== undefined);\n\n  if (signErrors.length || verifyErrors.length) {\n    throw new TransactionSigningError(signErrors, verifyErrors);\n  }\n\n  return signedTransaction;\n}\n"]}
|