utxo-lib 1.0.9 → 1.1.0
Sign up to get free protection for your applications and to get access to all the features.
- package/README.md +19 -16
- package/dist/src/address.d.ts.map +1 -1
- package/dist/src/address.js +11 -1
- package/dist/src/addressFormat.d.ts +1 -1
- package/dist/src/addressFormat.d.ts.map +1 -1
- package/dist/src/addressFormat.js +1 -1
- package/dist/src/bitgo/Musig2.d.ts +115 -17
- package/dist/src/bitgo/Musig2.d.ts.map +1 -1
- package/dist/src/bitgo/Musig2.js +283 -100
- package/dist/src/bitgo/PsbtUtil.d.ts +59 -0
- package/dist/src/bitgo/PsbtUtil.d.ts.map +1 -0
- package/dist/src/bitgo/PsbtUtil.js +91 -0
- package/dist/src/bitgo/UtxoPsbt.d.ts +180 -47
- package/dist/src/bitgo/UtxoPsbt.d.ts.map +1 -1
- package/dist/src/bitgo/UtxoPsbt.js +651 -107
- package/dist/src/bitgo/bitcoincash/address.js +2 -2
- package/dist/src/bitgo/index.d.ts +11 -0
- package/dist/src/bitgo/index.d.ts.map +1 -1
- package/dist/src/bitgo/index.js +6 -2
- package/dist/src/bitgo/legacysafe/index.d.ts +15 -0
- package/dist/src/bitgo/legacysafe/index.d.ts.map +1 -0
- package/dist/src/bitgo/legacysafe/index.js +61 -0
- package/dist/src/bitgo/litecoin/LitecoinPsbt.d.ts +10 -0
- package/dist/src/bitgo/litecoin/LitecoinPsbt.d.ts.map +1 -0
- package/dist/src/bitgo/litecoin/LitecoinPsbt.js +17 -0
- package/dist/src/bitgo/litecoin/LitecoinTransaction.d.ts +16 -0
- package/dist/src/bitgo/litecoin/LitecoinTransaction.d.ts.map +1 -0
- package/dist/src/bitgo/litecoin/LitecoinTransaction.js +46 -0
- package/dist/src/bitgo/litecoin/LitecoinTransactionBuilder.d.ts +10 -0
- package/dist/src/bitgo/litecoin/LitecoinTransactionBuilder.d.ts.map +1 -0
- package/dist/src/bitgo/litecoin/LitecoinTransactionBuilder.js +15 -0
- package/dist/src/bitgo/litecoin/index.d.ts +4 -0
- package/dist/src/bitgo/litecoin/index.d.ts.map +1 -0
- package/dist/src/bitgo/litecoin/index.js +16 -0
- package/dist/src/bitgo/outputScripts.d.ts +3 -1
- package/dist/src/bitgo/outputScripts.d.ts.map +1 -1
- package/dist/src/bitgo/outputScripts.js +18 -10
- package/dist/src/bitgo/parseInput.d.ts +49 -20
- package/dist/src/bitgo/parseInput.d.ts.map +1 -1
- package/dist/src/bitgo/parseInput.js +109 -25
- package/dist/src/bitgo/psbt/fromHalfSigned.d.ts.map +1 -1
- package/dist/src/bitgo/psbt/fromHalfSigned.js +9 -6
- package/dist/src/bitgo/signature.d.ts +3 -3
- package/dist/src/bitgo/signature.d.ts.map +1 -1
- package/dist/src/bitgo/signature.js +48 -16
- package/dist/src/bitgo/transaction.d.ts +18 -3
- package/dist/src/bitgo/transaction.d.ts.map +1 -1
- package/dist/src/bitgo/transaction.js +28 -15
- package/dist/src/bitgo/types.d.ts +2 -0
- package/dist/src/bitgo/types.d.ts.map +1 -1
- package/dist/src/bitgo/types.js +1 -1
- package/dist/src/bitgo/wallet/Psbt.d.ts +104 -12
- package/dist/src/bitgo/wallet/Psbt.d.ts.map +1 -1
- package/dist/src/bitgo/wallet/Psbt.js +285 -70
- package/dist/src/bitgo/wallet/Unspent.d.ts +28 -0
- package/dist/src/bitgo/wallet/Unspent.d.ts.map +1 -1
- package/dist/src/bitgo/wallet/Unspent.js +172 -68
- package/dist/src/bitgo/wallet/WalletOutput.d.ts +17 -1
- package/dist/src/bitgo/wallet/WalletOutput.d.ts.map +1 -1
- package/dist/src/bitgo/wallet/WalletOutput.js +64 -23
- package/dist/src/bitgo/wallet/chains.d.ts +2 -2
- package/dist/src/bitgo/wallet/chains.d.ts.map +1 -1
- package/dist/src/bitgo/wallet/chains.js +1 -1
- package/dist/src/bitgo/zcash/ZcashPsbt.d.ts +0 -1
- package/dist/src/bitgo/zcash/ZcashPsbt.d.ts.map +1 -1
- package/dist/src/bitgo/zcash/ZcashPsbt.js +6 -14
- package/dist/src/musig.d.ts +0 -1
- package/dist/src/musig.d.ts.map +1 -1
- package/dist/src/musig.js +15 -29
- package/dist/src/networks.d.ts +1 -2
- package/dist/src/networks.d.ts.map +1 -1
- package/dist/src/networks.js +22 -29
- package/dist/src/noble_ecc.d.ts.map +1 -1
- package/dist/src/noble_ecc.js +7 -3
- package/dist/src/payments/p2tr.d.ts.map +1 -1
- package/dist/src/payments/p2tr.js +15 -9
- package/dist/src/taproot.d.ts +16 -0
- package/dist/src/taproot.d.ts.map +1 -1
- package/dist/src/taproot.js +44 -2
- package/dist/src/testutil/index.d.ts +2 -0
- package/dist/src/testutil/index.d.ts.map +1 -1
- package/dist/src/testutil/index.js +3 -1
- package/dist/src/testutil/keys.d.ts +3 -0
- package/dist/src/testutil/keys.d.ts.map +1 -1
- package/dist/src/testutil/keys.js +17 -2
- package/dist/src/testutil/mock.d.ts +1 -1
- package/dist/src/testutil/mock.d.ts.map +1 -1
- package/dist/src/testutil/mock.js +12 -4
- package/dist/src/testutil/psbt.d.ts +89 -0
- package/dist/src/testutil/psbt.d.ts.map +1 -0
- package/dist/src/testutil/psbt.js +150 -0
- package/dist/src/testutil/transaction.d.ts +70 -0
- package/dist/src/testutil/transaction.d.ts.map +1 -0
- package/dist/src/testutil/transaction.js +107 -0
- package/package.json +6 -6
@@ -0,0 +1,150 @@
|
|
1
|
+
"use strict";
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
3
|
+
exports.verifyFullySignedSignatures = exports.constructPsbt = exports.signAllPsbtInputs = exports.signPsbtInput = exports.getSigners = exports.toUnspent = exports.outputScriptTypes = exports.inputScriptTypes = void 0;
|
4
|
+
const assert = require("assert");
|
5
|
+
const outputScripts_1 = require("../bitgo/outputScripts");
|
6
|
+
const bitgo_1 = require("../bitgo");
|
7
|
+
const mock_1 = require("./mock");
|
8
|
+
const address_1 = require("../address");
|
9
|
+
/**
|
10
|
+
* array of supported input script types.
|
11
|
+
* use p2trMusig2 for p2trMusig2 script path.
|
12
|
+
* use taprootKeyPathSpend for p2trMusig2 key path.
|
13
|
+
*/
|
14
|
+
exports.inputScriptTypes = [...outputScripts_1.scriptTypes2Of3, 'taprootKeyPathSpend', outputScripts_1.scriptTypeP2shP2pk];
|
15
|
+
/**
|
16
|
+
* array of supported output script types.
|
17
|
+
*/
|
18
|
+
exports.outputScriptTypes = outputScripts_1.scriptTypes2Of3;
|
19
|
+
/**
|
20
|
+
* create unspent object from input script type, index, network and root wallet key.
|
21
|
+
*/
|
22
|
+
function toUnspent(input, index, network, rootWalletKeys) {
|
23
|
+
if (input.scriptType === 'p2shP2pk') {
|
24
|
+
return mock_1.mockReplayProtectionUnspent(network, input.value, { key: rootWalletKeys['user'], vout: index });
|
25
|
+
}
|
26
|
+
else {
|
27
|
+
const chain = bitgo_1.getInternalChainCode(input.scriptType === 'taprootKeyPathSpend' ? 'p2trMusig2' : input.scriptType);
|
28
|
+
return mock_1.mockWalletUnspent(network, input.value, {
|
29
|
+
chain,
|
30
|
+
vout: index,
|
31
|
+
keys: rootWalletKeys,
|
32
|
+
index,
|
33
|
+
});
|
34
|
+
}
|
35
|
+
}
|
36
|
+
exports.toUnspent = toUnspent;
|
37
|
+
/**
|
38
|
+
* returns signer and cosigner names for InputScriptType.
|
39
|
+
* user and undefined as signer and cosigner respectively for p2shP2pk.
|
40
|
+
* user and backup as signer and cosigner respectively for p2trMusig2.
|
41
|
+
* user and bitgo as signer and cosigner respectively for other input script types.
|
42
|
+
*/
|
43
|
+
function getSigners(inputType) {
|
44
|
+
return {
|
45
|
+
signerName: 'user',
|
46
|
+
cosignerName: inputType === 'p2shP2pk' ? undefined : inputType === 'p2trMusig2' ? 'backup' : 'bitgo',
|
47
|
+
};
|
48
|
+
}
|
49
|
+
exports.getSigners = getSigners;
|
50
|
+
/**
|
51
|
+
* signs with first or second signature for single input.
|
52
|
+
* p2shP2pk is signed only with first sign.
|
53
|
+
*/
|
54
|
+
function signPsbtInput(psbt, input, inputIndex, rootWalletKeys, sign, params) {
|
55
|
+
const { signers, deterministic } = params ?? {};
|
56
|
+
const { signerName, cosignerName } = signers ? signers : getSigners(input.scriptType);
|
57
|
+
if (sign === 'halfsigned') {
|
58
|
+
if (input.scriptType === 'p2shP2pk') {
|
59
|
+
psbt.signInput(inputIndex, rootWalletKeys[signerName]);
|
60
|
+
}
|
61
|
+
else {
|
62
|
+
psbt.signInputHD(inputIndex, rootWalletKeys[signerName]);
|
63
|
+
}
|
64
|
+
}
|
65
|
+
if (sign === 'fullsigned' && cosignerName && input.scriptType !== 'p2shP2pk') {
|
66
|
+
psbt.signInputHD(inputIndex, rootWalletKeys[cosignerName], { deterministic });
|
67
|
+
}
|
68
|
+
}
|
69
|
+
exports.signPsbtInput = signPsbtInput;
|
70
|
+
/**
|
71
|
+
* signs with first or second signature for all inputs.
|
72
|
+
* p2shP2pk is signed only with first sign.
|
73
|
+
*/
|
74
|
+
function signAllPsbtInputs(psbt, inputs, rootWalletKeys, sign, params) {
|
75
|
+
const { signers, deterministic } = params ?? {};
|
76
|
+
inputs.forEach((input, inputIndex) => {
|
77
|
+
signPsbtInput(psbt, input, inputIndex, rootWalletKeys, sign, { signers, deterministic });
|
78
|
+
});
|
79
|
+
}
|
80
|
+
exports.signAllPsbtInputs = signAllPsbtInputs;
|
81
|
+
/**
|
82
|
+
* construct psbt for given inputs, outputs, network and root wallet keys.
|
83
|
+
*/
|
84
|
+
function constructPsbt(inputs, outputs, network, rootWalletKeys, sign, params) {
|
85
|
+
const { signers, deterministic } = params ?? {};
|
86
|
+
const totalInputAmount = inputs.reduce((sum, input) => sum + input.value, BigInt(0));
|
87
|
+
const outputInputAmount = outputs.reduce((sum, output) => sum + output.value, BigInt(0));
|
88
|
+
assert(totalInputAmount >= outputInputAmount, 'total output can not exceed total input');
|
89
|
+
assert(!outputs.some((o) => (o.scriptType && o.address) || (!o.scriptType && !o.address)), 'only either output script type or address should be provided');
|
90
|
+
const psbt = bitgo_1.createPsbtForNetwork({ network });
|
91
|
+
const unspents = inputs.map((input, i) => toUnspent(input, i, network, rootWalletKeys));
|
92
|
+
unspents.forEach((u, i) => {
|
93
|
+
const { signerName, cosignerName } = signers ? signers : getSigners(inputs[i].scriptType);
|
94
|
+
if (bitgo_1.isWalletUnspent(u) && cosignerName) {
|
95
|
+
bitgo_1.addWalletUnspentToPsbt(psbt, u, rootWalletKeys, signerName, cosignerName);
|
96
|
+
}
|
97
|
+
else {
|
98
|
+
const { redeemScript } = outputScripts_1.createOutputScriptP2shP2pk(rootWalletKeys[signerName].publicKey);
|
99
|
+
assert(redeemScript);
|
100
|
+
bitgo_1.addReplayProtectionUnspentToPsbt(psbt, u, redeemScript);
|
101
|
+
}
|
102
|
+
});
|
103
|
+
outputs.forEach((output, i) => {
|
104
|
+
if (output.scriptType) {
|
105
|
+
bitgo_1.addWalletOutputToPsbt(psbt, rootWalletKeys, output.isInternalAddress ? bitgo_1.getInternalChainCode(output.scriptType) : bitgo_1.getExternalChainCode(output.scriptType), i, output.value);
|
106
|
+
}
|
107
|
+
else if (output.address) {
|
108
|
+
const { address, value } = output;
|
109
|
+
psbt.addOutput({ script: address_1.toOutputScript(address, network), value });
|
110
|
+
}
|
111
|
+
});
|
112
|
+
if (sign === 'unsigned') {
|
113
|
+
return psbt;
|
114
|
+
}
|
115
|
+
psbt.setAllInputsMusig2NonceHD(rootWalletKeys['user']);
|
116
|
+
psbt.setAllInputsMusig2NonceHD(rootWalletKeys['bitgo'], { deterministic });
|
117
|
+
signAllPsbtInputs(psbt, inputs, rootWalletKeys, 'halfsigned', { signers });
|
118
|
+
if (sign === 'fullsigned') {
|
119
|
+
signAllPsbtInputs(psbt, inputs, rootWalletKeys, sign, { signers, deterministic });
|
120
|
+
}
|
121
|
+
return psbt;
|
122
|
+
}
|
123
|
+
exports.constructPsbt = constructPsbt;
|
124
|
+
/**
|
125
|
+
* Verifies signatures of fully signed tx (with taproot key path support).
|
126
|
+
* NOTE: taproot key path tx can only be built and signed with PSBT.
|
127
|
+
*/
|
128
|
+
function verifyFullySignedSignatures(tx, unspents, walletKeys, signer, cosigner) {
|
129
|
+
const prevOutputs = unspents.map((u) => bitgo_1.toOutput(u, tx.network));
|
130
|
+
return unspents.every((u, index) => {
|
131
|
+
if (bitgo_1.parseSignatureScript2Of3(tx.ins[index]).scriptType === 'taprootKeyPathSpend') {
|
132
|
+
const result = bitgo_1.getSignatureVerifications(tx, index, u.value, undefined, prevOutputs);
|
133
|
+
return result.length === 1 && result[0].signature;
|
134
|
+
}
|
135
|
+
else {
|
136
|
+
const result = bitgo_1.verifySignatureWithUnspent(tx, index, unspents, walletKeys);
|
137
|
+
if ((signer === 'user' && cosigner === 'bitgo') || (signer === 'bitgo' && cosigner === 'user')) {
|
138
|
+
return result[0] && !result[1] && result[2];
|
139
|
+
}
|
140
|
+
else if ((signer === 'user' && cosigner === 'backup') || (signer === 'backup' && cosigner === 'user')) {
|
141
|
+
return result[0] && result[1] && !result[2];
|
142
|
+
}
|
143
|
+
else {
|
144
|
+
return !result[0] && result[1] && result[2];
|
145
|
+
}
|
146
|
+
}
|
147
|
+
});
|
148
|
+
}
|
149
|
+
exports.verifyFullySignedSignatures = verifyFullySignedSignatures;
|
150
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"psbt.js","sourceRoot":"","sources":["../../../src/testutil/psbt.ts"],"names":[],"mappings":"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","sourcesContent":["import * as assert from 'assert';\n\nimport {\n  createOutputScriptP2shP2pk,\n  ScriptType,\n  ScriptType2Of3,\n  scriptTypeP2shP2pk,\n  scriptTypes2Of3,\n} from '../bitgo/outputScripts';\nimport {\n  addReplayProtectionUnspentToPsbt,\n  addWalletOutputToPsbt,\n  addWalletUnspentToPsbt,\n  createPsbtForNetwork,\n  getExternalChainCode,\n  getInternalChainCode,\n  getSignatureVerifications,\n  isWalletUnspent,\n  KeyName,\n  parseSignatureScript2Of3,\n  RootWalletKeys,\n  toOutput,\n  Unspent,\n  UtxoPsbt,\n  UtxoTransaction,\n  verifySignatureWithUnspent,\n} from '../bitgo';\nimport { Network } from '../networks';\nimport { mockReplayProtectionUnspent, mockWalletUnspent } from './mock';\nimport { toOutputScript } from '../address';\n\n/**\n * input script type and value.\n * use p2trMusig2 for p2trMusig2 script path.\n * use taprootKeyPathSpend for p2trMusig2 key path.\n */\nexport type InputScriptType = ScriptType | 'taprootKeyPathSpend';\nexport type OutputScriptType = ScriptType2Of3;\n\n/**\n * input script type and value\n */\nexport interface Input {\n  scriptType: InputScriptType;\n  value: bigint;\n}\n\n/**\n * should set either address or scriptType, never both.\n * set isInternalAddress=true for internal output address\n */\nexport interface Output {\n  address?: string;\n  scriptType?: OutputScriptType;\n  value: bigint;\n  isInternalAddress?: boolean;\n}\n\n/**\n * array of supported input script types.\n * use p2trMusig2 for p2trMusig2 script path.\n * use taprootKeyPathSpend for p2trMusig2 key path.\n */\nexport const inputScriptTypes = [...scriptTypes2Of3, 'taprootKeyPathSpend', scriptTypeP2shP2pk] as const;\n\n/**\n * array of supported output script types.\n */\nexport const outputScriptTypes = scriptTypes2Of3;\n\n/**\n * create unspent object from input script type, index, network and root wallet key.\n */\nexport function toUnspent(\n  input: Input,\n  index: number,\n  network: Network,\n  rootWalletKeys: RootWalletKeys\n): Unspent<bigint> {\n  if (input.scriptType === 'p2shP2pk') {\n    return mockReplayProtectionUnspent(network, input.value, { key: rootWalletKeys['user'], vout: index });\n  } else {\n    const chain = getInternalChainCode(input.scriptType === 'taprootKeyPathSpend' ? 'p2trMusig2' : input.scriptType);\n    return mockWalletUnspent(network, input.value, {\n      chain,\n      vout: index,\n      keys: rootWalletKeys,\n      index,\n    });\n  }\n}\n\n/**\n * returns signer and cosigner names for InputScriptType.\n * user and undefined as signer and cosigner respectively for p2shP2pk.\n * user and backup as signer and cosigner respectively for p2trMusig2.\n * user and bitgo as signer and cosigner respectively for other input script types.\n */\nexport function getSigners(inputType: InputScriptType): { signerName: KeyName; cosignerName?: KeyName } {\n  return {\n    signerName: 'user',\n    cosignerName: inputType === 'p2shP2pk' ? undefined : inputType === 'p2trMusig2' ? 'backup' : 'bitgo',\n  };\n}\n\n/**\n * signs with first or second signature for single input.\n * p2shP2pk is signed only with first sign.\n */\nexport function signPsbtInput(\n  psbt: UtxoPsbt,\n  input: Input,\n  inputIndex: number,\n  rootWalletKeys: RootWalletKeys,\n  sign: 'halfsigned' | 'fullsigned',\n  params?: {\n    signers?: { signerName: KeyName; cosignerName?: KeyName };\n    deterministic?: boolean;\n  }\n): void {\n  const { signers, deterministic } = params ?? {};\n  const { signerName, cosignerName } = signers ? signers : getSigners(input.scriptType);\n  if (sign === 'halfsigned') {\n    if (input.scriptType === 'p2shP2pk') {\n      psbt.signInput(inputIndex, rootWalletKeys[signerName]);\n    } else {\n      psbt.signInputHD(inputIndex, rootWalletKeys[signerName]);\n    }\n  }\n  if (sign === 'fullsigned' && cosignerName && input.scriptType !== 'p2shP2pk') {\n    psbt.signInputHD(inputIndex, rootWalletKeys[cosignerName], { deterministic });\n  }\n}\n\n/**\n * signs with first or second signature for all inputs.\n * p2shP2pk is signed only with first sign.\n */\nexport function signAllPsbtInputs(\n  psbt: UtxoPsbt,\n  inputs: Input[],\n  rootWalletKeys: RootWalletKeys,\n  sign: 'halfsigned' | 'fullsigned',\n  params?: {\n    signers?: { signerName: KeyName; cosignerName?: KeyName };\n    deterministic?: boolean;\n  }\n): void {\n  const { signers, deterministic } = params ?? {};\n  inputs.forEach((input, inputIndex) => {\n    signPsbtInput(psbt, input, inputIndex, rootWalletKeys, sign, { signers, deterministic });\n  });\n}\n\n/**\n * construct psbt for given inputs, outputs, network and root wallet keys.\n */\nexport function constructPsbt(\n  inputs: Input[],\n  outputs: Output[],\n  network: Network,\n  rootWalletKeys: RootWalletKeys,\n  sign: 'unsigned' | 'halfsigned' | 'fullsigned',\n  params?: {\n    signers?: { signerName: KeyName; cosignerName?: KeyName };\n    deterministic?: boolean;\n  }\n): UtxoPsbt {\n  const { signers, deterministic } = params ?? {};\n  const totalInputAmount = inputs.reduce((sum, input) => sum + input.value, BigInt(0));\n  const outputInputAmount = outputs.reduce((sum, output) => sum + output.value, BigInt(0));\n  assert(totalInputAmount >= outputInputAmount, 'total output can not exceed total input');\n  assert(\n    !outputs.some((o) => (o.scriptType && o.address) || (!o.scriptType && !o.address)),\n    'only either output script type or address should be provided'\n  );\n\n  const psbt = createPsbtForNetwork({ network });\n  const unspents = inputs.map((input, i) => toUnspent(input, i, network, rootWalletKeys));\n\n  unspents.forEach((u, i) => {\n    const { signerName, cosignerName } = signers ? signers : getSigners(inputs[i].scriptType);\n    if (isWalletUnspent(u) && cosignerName) {\n      addWalletUnspentToPsbt(psbt, u, rootWalletKeys, signerName, cosignerName);\n    } else {\n      const { redeemScript } = createOutputScriptP2shP2pk(rootWalletKeys[signerName].publicKey);\n      assert(redeemScript);\n      addReplayProtectionUnspentToPsbt(psbt, u, redeemScript);\n    }\n  });\n\n  outputs.forEach((output, i) => {\n    if (output.scriptType) {\n      addWalletOutputToPsbt(\n        psbt,\n        rootWalletKeys,\n        output.isInternalAddress ? getInternalChainCode(output.scriptType) : getExternalChainCode(output.scriptType),\n        i,\n        output.value\n      );\n    } else if (output.address) {\n      const { address, value } = output;\n      psbt.addOutput({ script: toOutputScript(address, network), value });\n    }\n  });\n\n  if (sign === 'unsigned') {\n    return psbt;\n  }\n\n  psbt.setAllInputsMusig2NonceHD(rootWalletKeys['user']);\n  psbt.setAllInputsMusig2NonceHD(rootWalletKeys['bitgo'], { deterministic });\n\n  signAllPsbtInputs(psbt, inputs, rootWalletKeys, 'halfsigned', { signers });\n\n  if (sign === 'fullsigned') {\n    signAllPsbtInputs(psbt, inputs, rootWalletKeys, sign, { signers, deterministic });\n  }\n\n  return psbt;\n}\n\n/**\n * Verifies signatures of fully signed tx (with taproot key path support).\n * NOTE: taproot key path tx can only be built and signed with PSBT.\n */\nexport function verifyFullySignedSignatures(\n  tx: UtxoTransaction<bigint>,\n  unspents: Unspent<bigint>[],\n  walletKeys: RootWalletKeys,\n  signer: KeyName,\n  cosigner: KeyName\n): boolean {\n  const prevOutputs = unspents.map((u) => toOutput(u, tx.network));\n  return unspents.every((u, index) => {\n    if (parseSignatureScript2Of3(tx.ins[index]).scriptType === 'taprootKeyPathSpend') {\n      const result = getSignatureVerifications(tx, index, u.value, undefined, prevOutputs);\n      return result.length === 1 && result[0].signature;\n    } else {\n      const result = verifySignatureWithUnspent(tx, index, unspents, walletKeys);\n      if ((signer === 'user' && cosigner === 'bitgo') || (signer === 'bitgo' && cosigner === 'user')) {\n        return result[0] && !result[1] && result[2];\n      } else if ((signer === 'user' && cosigner === 'backup') || (signer === 'backup' && cosigner === 'user')) {\n        return result[0] && result[1] && !result[2];\n      } else {\n        return !result[0] && result[1] && result[2];\n      }\n    }\n  });\n}\n"]}
|
@@ -0,0 +1,70 @@
|
|
1
|
+
import { ScriptType, ScriptType2Of3 } from '../bitgo/outputScripts';
|
2
|
+
import { KeyName, RootWalletKeys, Unspent, UtxoTransactionBuilder } from '../bitgo';
|
3
|
+
import { Network } from '../networks';
|
4
|
+
/**
|
5
|
+
* input script type and value.
|
6
|
+
*/
|
7
|
+
export declare type TxnInputScriptType = Exclude<ScriptType, 'p2trMusig2'>;
|
8
|
+
export declare type TxnOutputScriptType = ScriptType2Of3;
|
9
|
+
/**
|
10
|
+
* output script type and value
|
11
|
+
*/
|
12
|
+
export interface TxnInput<TNumber extends number | bigint> {
|
13
|
+
scriptType: TxnInputScriptType;
|
14
|
+
value: TNumber;
|
15
|
+
}
|
16
|
+
/**
|
17
|
+
* should set either address or scriptType, never both.
|
18
|
+
* set isInternalAddress=true for internal output address
|
19
|
+
*/
|
20
|
+
export interface TxnOutput<TNumber extends number | bigint> {
|
21
|
+
address?: string;
|
22
|
+
scriptType?: TxnOutputScriptType;
|
23
|
+
value: TNumber;
|
24
|
+
isInternalAddress?: boolean;
|
25
|
+
}
|
26
|
+
/**
|
27
|
+
* array of supported input script types.
|
28
|
+
*/
|
29
|
+
export declare const txnInputScriptTypes: readonly ["p2sh", "p2shP2wsh", "p2wsh", "p2tr", "p2shP2pk"];
|
30
|
+
/**
|
31
|
+
* array of supported output script types.
|
32
|
+
*/
|
33
|
+
export declare const txnOutputScriptTypes: readonly ["p2sh", "p2shP2wsh", "p2wsh", "p2tr", "p2trMusig2"];
|
34
|
+
/**
|
35
|
+
* create unspent object from input script type, index, network and root wallet key.
|
36
|
+
*/
|
37
|
+
export declare function toTxnUnspent<TNumber extends number | bigint>(input: TxnInput<TNumber>, index: number, network: Network, rootWalletKeys: RootWalletKeys): Unspent<TNumber>;
|
38
|
+
/**
|
39
|
+
* returns signer and cosigner names for TxnInputScriptType.
|
40
|
+
* user and undefined as signer and cosigner respectively for p2shP2pk.
|
41
|
+
* user and bitgo as signer and cosigner respectively for other input script types.
|
42
|
+
*/
|
43
|
+
export declare function getTxnSigners(inputType: TxnInputScriptType): {
|
44
|
+
signerName: KeyName;
|
45
|
+
cosignerName?: KeyName;
|
46
|
+
};
|
47
|
+
/**
|
48
|
+
* signs with first or second signature for single input.
|
49
|
+
* p2shP2pk is signed only with first sign.
|
50
|
+
*/
|
51
|
+
export declare function signTxnInput<TNumber extends number | bigint>(txb: UtxoTransactionBuilder<TNumber>, input: TxnInput<TNumber>, inputIndex: number, rootWalletKeys: RootWalletKeys, sign: 'halfsigned' | 'fullsigned', signers?: {
|
52
|
+
signerName: KeyName;
|
53
|
+
cosignerName?: KeyName;
|
54
|
+
}): void;
|
55
|
+
/**
|
56
|
+
* signs with first or second signature for all inputs.
|
57
|
+
* p2shP2pk is signed only with first sign.
|
58
|
+
*/
|
59
|
+
export declare function signAllTxnInputs<TNumber extends number | bigint>(txb: UtxoTransactionBuilder<TNumber>, inputs: TxnInput<TNumber>[], rootWalletKeys: RootWalletKeys, sign: 'halfsigned' | 'fullsigned', signers?: {
|
60
|
+
signerName: KeyName;
|
61
|
+
cosignerName?: KeyName;
|
62
|
+
}): void;
|
63
|
+
/**
|
64
|
+
* construct transaction for given inputs, outputs, network and root wallet keys.
|
65
|
+
*/
|
66
|
+
export declare function constructTxnBuilder<TNumber extends number | bigint>(inputs: TxnInput<TNumber>[], outputs: TxnOutput<TNumber>[], network: Network, rootWalletKeys: RootWalletKeys, sign: 'unsigned' | 'halfsigned' | 'fullsigned', signers?: {
|
67
|
+
signerName: KeyName;
|
68
|
+
cosignerName?: KeyName;
|
69
|
+
}): UtxoTransactionBuilder<TNumber>;
|
70
|
+
//# sourceMappingURL=transaction.d.ts.map
|
@@ -0,0 +1 @@
|
|
1
|
+
{"version":3,"file":"transaction.d.ts","sourceRoot":"","sources":["../../../src/testutil/transaction.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,UAAU,EAAE,cAAc,EAAuC,MAAM,wBAAwB,CAAC;AACzG,OAAO,EAGL,OAAO,EAEP,cAAc,EACd,OAAO,EACP,sBAAsB,EAOvB,MAAM,UAAU,CAAC;AAClB,OAAO,EAAE,OAAO,EAAE,MAAM,aAAa,CAAC;AAGtC;;GAEG;AACH,oBAAY,kBAAkB,GAAG,OAAO,CAAC,UAAU,EAAE,YAAY,CAAC,CAAC;AACnE,oBAAY,mBAAmB,GAAG,cAAc,CAAC;AAEjD;;GAEG;AACH,MAAM,WAAW,QAAQ,CAAC,OAAO,SAAS,MAAM,GAAG,MAAM;IACvD,UAAU,EAAE,kBAAkB,CAAC;IAC/B,KAAK,EAAE,OAAO,CAAC;CAChB;AAED;;;GAGG;AACH,MAAM,WAAW,SAAS,CAAC,OAAO,SAAS,MAAM,GAAG,MAAM;IACxD,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,UAAU,CAAC,EAAE,mBAAmB,CAAC;IACjC,KAAK,EAAE,OAAO,CAAC;IACf,iBAAiB,CAAC,EAAE,OAAO,CAAC;CAC7B;AAED;;GAEG;AACH,eAAO,MAAM,mBAAmB,6DAAsE,CAAC;AAEvG;;GAEG;AACH,eAAO,MAAM,oBAAoB,+DAAkB,CAAC;AAEpD;;GAEG;AACH,wBAAgB,YAAY,CAAC,OAAO,SAAS,MAAM,GAAG,MAAM,EAC1D,KAAK,EAAE,QAAQ,CAAC,OAAO,CAAC,EACxB,KAAK,EAAE,MAAM,EACb,OAAO,EAAE,OAAO,EAChB,cAAc,EAAE,cAAc,GAC7B,OAAO,CAAC,OAAO,CAAC,CAWlB;AAED;;;;GAIG;AACH,wBAAgB,aAAa,CAAC,SAAS,EAAE,kBAAkB,GAAG;IAAE,UAAU,EAAE,OAAO,CAAC;IAAC,YAAY,CAAC,EAAE,OAAO,CAAA;CAAE,CAK5G;AAED;;;GAGG;AACH,wBAAgB,YAAY,CAAC,OAAO,SAAS,MAAM,GAAG,MAAM,EAC1D,GAAG,EAAE,sBAAsB,CAAC,OAAO,CAAC,EACpC,KAAK,EAAE,QAAQ,CAAC,OAAO,CAAC,EACxB,UAAU,EAAE,MAAM,EAClB,cAAc,EAAE,cAAc,EAC9B,IAAI,EAAE,YAAY,GAAG,YAAY,EACjC,OAAO,CAAC,EAAE;IAAE,UAAU,EAAE,OAAO,CAAC;IAAC,YAAY,CAAC,EAAE,OAAO,CAAA;CAAE,GACxD,IAAI,CAuBN;AAED;;;GAGG;AACH,wBAAgB,gBAAgB,CAAC,OAAO,SAAS,MAAM,GAAG,MAAM,EAC9D,GAAG,EAAE,sBAAsB,CAAC,OAAO,CAAC,EACpC,MAAM,EAAE,QAAQ,CAAC,OAAO,CAAC,EAAE,EAC3B,cAAc,EAAE,cAAc,EAC9B,IAAI,EAAE,YAAY,GAAG,YAAY,EACjC,OAAO,CAAC,EAAE;IAAE,UAAU,EAAE,OAAO,CAAC;IAAC,YAAY,CAAC,EAAE,OAAO,CAAA;CAAE,GACxD,IAAI,CAIN;AAED;;GAEG;AACH,wBAAgB,mBAAmB,CAAC,OAAO,SAAS,MAAM,GAAG,MAAM,EACjE,MAAM,EAAE,QAAQ,CAAC,OAAO,CAAC,EAAE,EAC3B,OAAO,EAAE,SAAS,CAAC,OAAO,CAAC,EAAE,EAC7B,OAAO,EAAE,OAAO,EAChB,cAAc,EAAE,cAAc,EAC9B,IAAI,EAAE,UAAU,GAAG,YAAY,GAAG,YAAY,EAC9C,OAAO,CAAC,EAAE;IAAE,UAAU,EAAE,OAAO,CAAC;IAAC,YAAY,CAAC,EAAE,OAAO,CAAA;CAAE,GACxD,sBAAsB,CAAC,OAAO,CAAC,CA2CjC"}
|
@@ -0,0 +1,107 @@
|
|
1
|
+
"use strict";
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
3
|
+
exports.constructTxnBuilder = exports.signAllTxnInputs = exports.signTxnInput = exports.getTxnSigners = exports.toTxnUnspent = exports.txnOutputScriptTypes = exports.txnInputScriptTypes = void 0;
|
4
|
+
const assert = require("assert");
|
5
|
+
const outputScripts_1 = require("../bitgo/outputScripts");
|
6
|
+
const bitgo_1 = require("../bitgo");
|
7
|
+
const mock_1 = require("./mock");
|
8
|
+
/**
|
9
|
+
* array of supported input script types.
|
10
|
+
*/
|
11
|
+
exports.txnInputScriptTypes = ['p2sh', 'p2shP2wsh', 'p2wsh', 'p2tr', outputScripts_1.scriptTypeP2shP2pk];
|
12
|
+
/**
|
13
|
+
* array of supported output script types.
|
14
|
+
*/
|
15
|
+
exports.txnOutputScriptTypes = outputScripts_1.scriptTypes2Of3;
|
16
|
+
/**
|
17
|
+
* create unspent object from input script type, index, network and root wallet key.
|
18
|
+
*/
|
19
|
+
function toTxnUnspent(input, index, network, rootWalletKeys) {
|
20
|
+
if (input.scriptType === 'p2shP2pk') {
|
21
|
+
return mock_1.mockReplayProtectionUnspent(network, input.value, { key: rootWalletKeys['user'], vout: index });
|
22
|
+
}
|
23
|
+
else {
|
24
|
+
return mock_1.mockWalletUnspent(network, input.value, {
|
25
|
+
chain: bitgo_1.getInternalChainCode(input.scriptType),
|
26
|
+
vout: index,
|
27
|
+
keys: rootWalletKeys,
|
28
|
+
index,
|
29
|
+
});
|
30
|
+
}
|
31
|
+
}
|
32
|
+
exports.toTxnUnspent = toTxnUnspent;
|
33
|
+
/**
|
34
|
+
* returns signer and cosigner names for TxnInputScriptType.
|
35
|
+
* user and undefined as signer and cosigner respectively for p2shP2pk.
|
36
|
+
* user and bitgo as signer and cosigner respectively for other input script types.
|
37
|
+
*/
|
38
|
+
function getTxnSigners(inputType) {
|
39
|
+
return {
|
40
|
+
signerName: 'user',
|
41
|
+
cosignerName: inputType === 'p2shP2pk' ? undefined : 'bitgo',
|
42
|
+
};
|
43
|
+
}
|
44
|
+
exports.getTxnSigners = getTxnSigners;
|
45
|
+
/**
|
46
|
+
* signs with first or second signature for single input.
|
47
|
+
* p2shP2pk is signed only with first sign.
|
48
|
+
*/
|
49
|
+
function signTxnInput(txb, input, inputIndex, rootWalletKeys, sign, signers) {
|
50
|
+
const { signerName, cosignerName } = signers ? signers : getTxnSigners(input.scriptType);
|
51
|
+
const unspent = toTxnUnspent(input, inputIndex, txb.network, rootWalletKeys);
|
52
|
+
if (sign === 'halfsigned') {
|
53
|
+
if (input.scriptType === 'p2shP2pk') {
|
54
|
+
bitgo_1.signInputP2shP2pk(txb, inputIndex, rootWalletKeys[signerName]);
|
55
|
+
}
|
56
|
+
else if (bitgo_1.isWalletUnspent(unspent) && cosignerName) {
|
57
|
+
bitgo_1.signInputWithUnspent(txb, inputIndex, unspent, bitgo_1.WalletUnspentSigner.from(rootWalletKeys, rootWalletKeys[signerName], rootWalletKeys[cosignerName]));
|
58
|
+
}
|
59
|
+
}
|
60
|
+
if (bitgo_1.isWalletUnspent(unspent) && sign === 'fullsigned' && cosignerName) {
|
61
|
+
bitgo_1.signInputWithUnspent(txb, inputIndex, unspent, bitgo_1.WalletUnspentSigner.from(rootWalletKeys, rootWalletKeys[cosignerName], rootWalletKeys[signerName]));
|
62
|
+
}
|
63
|
+
}
|
64
|
+
exports.signTxnInput = signTxnInput;
|
65
|
+
/**
|
66
|
+
* signs with first or second signature for all inputs.
|
67
|
+
* p2shP2pk is signed only with first sign.
|
68
|
+
*/
|
69
|
+
function signAllTxnInputs(txb, inputs, rootWalletKeys, sign, signers) {
|
70
|
+
inputs.forEach((input, index) => {
|
71
|
+
signTxnInput(txb, input, index, rootWalletKeys, sign, signers);
|
72
|
+
});
|
73
|
+
}
|
74
|
+
exports.signAllTxnInputs = signAllTxnInputs;
|
75
|
+
/**
|
76
|
+
* construct transaction for given inputs, outputs, network and root wallet keys.
|
77
|
+
*/
|
78
|
+
function constructTxnBuilder(inputs, outputs, network, rootWalletKeys, sign, signers) {
|
79
|
+
const totalInputAmount = inputs.reduce((sum, input) => sum + BigInt(input.value), BigInt(0));
|
80
|
+
const outputInputAmount = outputs.reduce((sum, output) => sum + BigInt(output.value), BigInt(0));
|
81
|
+
assert(totalInputAmount >= outputInputAmount, 'total output can not exceed total input');
|
82
|
+
assert(!outputs.some((o) => (o.scriptType && o.address) || (!o.scriptType && !o.address)), 'only either output script type or address should be provided');
|
83
|
+
const txb = bitgo_1.createTransactionBuilderForNetwork(network);
|
84
|
+
const unspents = inputs.map((input, i) => toTxnUnspent(input, i, network, rootWalletKeys));
|
85
|
+
unspents.forEach((u, i) => {
|
86
|
+
bitgo_1.addToTransactionBuilder(txb, u);
|
87
|
+
});
|
88
|
+
outputs.forEach((output, i) => {
|
89
|
+
const address = output.scriptType
|
90
|
+
? bitgo_1.getWalletAddress(rootWalletKeys, output.isInternalAddress ? bitgo_1.getInternalChainCode(output.scriptType) : bitgo_1.getExternalChainCode(output.scriptType), i, network)
|
91
|
+
: output.address;
|
92
|
+
if (!address) {
|
93
|
+
throw new Error('address is missing');
|
94
|
+
}
|
95
|
+
txb.addOutput(address, output.value);
|
96
|
+
});
|
97
|
+
if (sign === 'unsigned') {
|
98
|
+
return txb;
|
99
|
+
}
|
100
|
+
signAllTxnInputs(txb, inputs, rootWalletKeys, 'halfsigned', signers);
|
101
|
+
if (sign === 'fullsigned') {
|
102
|
+
signAllTxnInputs(txb, inputs, rootWalletKeys, sign, signers);
|
103
|
+
}
|
104
|
+
return txb;
|
105
|
+
}
|
106
|
+
exports.constructTxnBuilder = constructTxnBuilder;
|
107
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"transaction.js","sourceRoot":"","sources":["../../../src/testutil/transaction.ts"],"names":[],"mappings":"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","sourcesContent":["import * as assert from 'assert';\n\nimport { ScriptType, ScriptType2Of3, scriptTypeP2shP2pk, scriptTypes2Of3 } from '../bitgo/outputScripts';\nimport {\n  getExternalChainCode,\n  isWalletUnspent,\n  KeyName,\n  getInternalChainCode,\n  RootWalletKeys,\n  Unspent,\n  UtxoTransactionBuilder,\n  createTransactionBuilderForNetwork,\n  addToTransactionBuilder,\n  getWalletAddress,\n  signInputP2shP2pk,\n  signInputWithUnspent,\n  WalletUnspentSigner,\n} from '../bitgo';\nimport { Network } from '../networks';\nimport { mockReplayProtectionUnspent, mockWalletUnspent } from './mock';\n\n/**\n * input script type and value.\n */\nexport type TxnInputScriptType = Exclude<ScriptType, 'p2trMusig2'>;\nexport type TxnOutputScriptType = ScriptType2Of3;\n\n/**\n * output script type and value\n */\nexport interface TxnInput<TNumber extends number | bigint> {\n  scriptType: TxnInputScriptType;\n  value: TNumber;\n}\n\n/**\n * should set either address or scriptType, never both.\n * set isInternalAddress=true for internal output address\n */\nexport interface TxnOutput<TNumber extends number | bigint> {\n  address?: string;\n  scriptType?: TxnOutputScriptType;\n  value: TNumber;\n  isInternalAddress?: boolean;\n}\n\n/**\n * array of supported input script types.\n */\nexport const txnInputScriptTypes = ['p2sh', 'p2shP2wsh', 'p2wsh', 'p2tr', scriptTypeP2shP2pk] as const;\n\n/**\n * array of supported output script types.\n */\nexport const txnOutputScriptTypes = scriptTypes2Of3;\n\n/**\n * create unspent object from input script type, index, network and root wallet key.\n */\nexport function toTxnUnspent<TNumber extends number | bigint>(\n  input: TxnInput<TNumber>,\n  index: number,\n  network: Network,\n  rootWalletKeys: RootWalletKeys\n): Unspent<TNumber> {\n  if (input.scriptType === 'p2shP2pk') {\n    return mockReplayProtectionUnspent<TNumber>(network, input.value, { key: rootWalletKeys['user'], vout: index });\n  } else {\n    return mockWalletUnspent<TNumber>(network, input.value, {\n      chain: getInternalChainCode(input.scriptType),\n      vout: index,\n      keys: rootWalletKeys,\n      index,\n    });\n  }\n}\n\n/**\n * returns signer and cosigner names for TxnInputScriptType.\n * user and undefined as signer and cosigner respectively for p2shP2pk.\n * user and bitgo as signer and cosigner respectively for other input script types.\n */\nexport function getTxnSigners(inputType: TxnInputScriptType): { signerName: KeyName; cosignerName?: KeyName } {\n  return {\n    signerName: 'user',\n    cosignerName: inputType === 'p2shP2pk' ? undefined : 'bitgo',\n  };\n}\n\n/**\n * signs with first or second signature for single input.\n * p2shP2pk is signed only with first sign.\n */\nexport function signTxnInput<TNumber extends number | bigint>(\n  txb: UtxoTransactionBuilder<TNumber>,\n  input: TxnInput<TNumber>,\n  inputIndex: number,\n  rootWalletKeys: RootWalletKeys,\n  sign: 'halfsigned' | 'fullsigned',\n  signers?: { signerName: KeyName; cosignerName?: KeyName }\n): void {\n  const { signerName, cosignerName } = signers ? signers : getTxnSigners(input.scriptType);\n  const unspent = toTxnUnspent(input, inputIndex, txb.network, rootWalletKeys);\n  if (sign === 'halfsigned') {\n    if (input.scriptType === 'p2shP2pk') {\n      signInputP2shP2pk(txb, inputIndex, rootWalletKeys[signerName]);\n    } else if (isWalletUnspent(unspent) && cosignerName) {\n      signInputWithUnspent(\n        txb,\n        inputIndex,\n        unspent,\n        WalletUnspentSigner.from(rootWalletKeys, rootWalletKeys[signerName], rootWalletKeys[cosignerName])\n      );\n    }\n  }\n  if (isWalletUnspent(unspent) && sign === 'fullsigned' && cosignerName) {\n    signInputWithUnspent(\n      txb,\n      inputIndex,\n      unspent,\n      WalletUnspentSigner.from(rootWalletKeys, rootWalletKeys[cosignerName], rootWalletKeys[signerName])\n    );\n  }\n}\n\n/**\n * signs with first or second signature for all inputs.\n * p2shP2pk is signed only with first sign.\n */\nexport function signAllTxnInputs<TNumber extends number | bigint>(\n  txb: UtxoTransactionBuilder<TNumber>,\n  inputs: TxnInput<TNumber>[],\n  rootWalletKeys: RootWalletKeys,\n  sign: 'halfsigned' | 'fullsigned',\n  signers?: { signerName: KeyName; cosignerName?: KeyName }\n): void {\n  inputs.forEach((input, index) => {\n    signTxnInput(txb, input, index, rootWalletKeys, sign, signers);\n  });\n}\n\n/**\n * construct transaction for given inputs, outputs, network and root wallet keys.\n */\nexport function constructTxnBuilder<TNumber extends number | bigint>(\n  inputs: TxnInput<TNumber>[],\n  outputs: TxnOutput<TNumber>[],\n  network: Network,\n  rootWalletKeys: RootWalletKeys,\n  sign: 'unsigned' | 'halfsigned' | 'fullsigned',\n  signers?: { signerName: KeyName; cosignerName?: KeyName }\n): UtxoTransactionBuilder<TNumber> {\n  const totalInputAmount = inputs.reduce((sum, input) => sum + BigInt(input.value), BigInt(0));\n  const outputInputAmount = outputs.reduce((sum, output) => sum + BigInt(output.value), BigInt(0));\n  assert(totalInputAmount >= outputInputAmount, 'total output can not exceed total input');\n  assert(\n    !outputs.some((o) => (o.scriptType && o.address) || (!o.scriptType && !o.address)),\n    'only either output script type or address should be provided'\n  );\n\n  const txb = createTransactionBuilderForNetwork<TNumber>(network);\n\n  const unspents = inputs.map((input, i) => toTxnUnspent(input, i, network, rootWalletKeys));\n\n  unspents.forEach((u, i) => {\n    addToTransactionBuilder(txb, u);\n  });\n\n  outputs.forEach((output, i) => {\n    const address = output.scriptType\n      ? getWalletAddress(\n          rootWalletKeys,\n          output.isInternalAddress ? getInternalChainCode(output.scriptType) : getExternalChainCode(output.scriptType),\n          i,\n          network\n        )\n      : output.address;\n    if (!address) {\n      throw new Error('address is missing');\n    }\n    txb.addOutput(address, output.value);\n  });\n\n  if (sign === 'unsigned') {\n    return txb;\n  }\n\n  signAllTxnInputs(txb, inputs, rootWalletKeys, 'halfsigned', signers);\n\n  if (sign === 'fullsigned') {\n    signAllTxnInputs(txb, inputs, rootWalletKeys, sign, signers);\n  }\n\n  return txb;\n}\n"]}
|
package/package.json
CHANGED
@@ -1,10 +1,10 @@
|
|
1
1
|
{
|
2
2
|
"name": "utxo-lib",
|
3
|
-
"version": "1.0
|
3
|
+
"version": "1.1.0",
|
4
4
|
"description": "Client-side Bitcoin JavaScript library",
|
5
5
|
"main": "./dist/src/index.js",
|
6
6
|
"engines": {
|
7
|
-
"node": ">=10.22.0 <
|
7
|
+
"node": ">=10.22.0 <19",
|
8
8
|
"npm": ">=3.10.10"
|
9
9
|
},
|
10
10
|
"keywords": [
|
@@ -46,13 +46,13 @@
|
|
46
46
|
"dist/src"
|
47
47
|
],
|
48
48
|
"dependencies": {
|
49
|
-
"@bitgo/blake2b": "^3.2.
|
49
|
+
"@bitgo/blake2b": "^3.2.4",
|
50
50
|
"@noble/secp256k1": "1.6.3",
|
51
51
|
"bech32": "^2.0.0",
|
52
|
-
"bip174": "npm:@bitgo-forks/bip174@3.1.0-master.
|
52
|
+
"bip174": "npm:@bitgo-forks/bip174@3.1.0-master.4",
|
53
53
|
"bip32": "^3.0.1",
|
54
54
|
"bitcoin-ops": "^1.3.0",
|
55
|
-
"bitcoinjs-lib": "npm:@bitgo-forks/bitcoinjs-lib@7.1.0-master.
|
55
|
+
"bitcoinjs-lib": "npm:@bitgo-forks/bitcoinjs-lib@7.1.0-master.6",
|
56
56
|
"bn.js": "^5.2.1",
|
57
57
|
"bs58check": "^2.1.2",
|
58
58
|
"cashaddress": "^1.1.0",
|
@@ -67,7 +67,7 @@
|
|
67
67
|
"devDependencies": {
|
68
68
|
"@types/elliptic": "^6.4.12",
|
69
69
|
"@types/fs-extra": "^9.0.12",
|
70
|
-
"@types/node": "^
|
70
|
+
"@types/node": "^16.18.46",
|
71
71
|
"axios": "^0.21.1",
|
72
72
|
"debug": "^3.1.0",
|
73
73
|
"fs-extra": "^9.1.0"
|