utxo-lib 1.0.8 → 1.1.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (104) hide show
  1. package/README.md +19 -16
  2. package/dist/src/address.d.ts.map +1 -1
  3. package/dist/src/address.js +11 -1
  4. package/dist/src/addressFormat.d.ts +1 -1
  5. package/dist/src/addressFormat.d.ts.map +1 -1
  6. package/dist/src/addressFormat.js +1 -1
  7. package/dist/src/base_crypto.d.ts +14 -0
  8. package/dist/src/base_crypto.d.ts.map +1 -0
  9. package/dist/src/base_crypto.js +215 -0
  10. package/dist/src/bitgo/Musig2.d.ts +115 -17
  11. package/dist/src/bitgo/Musig2.d.ts.map +1 -1
  12. package/dist/src/bitgo/Musig2.js +283 -101
  13. package/dist/src/bitgo/PsbtUtil.d.ts +59 -0
  14. package/dist/src/bitgo/PsbtUtil.d.ts.map +1 -0
  15. package/dist/src/bitgo/PsbtUtil.js +91 -0
  16. package/dist/src/bitgo/UtxoPsbt.d.ts +180 -47
  17. package/dist/src/bitgo/UtxoPsbt.d.ts.map +1 -1
  18. package/dist/src/bitgo/UtxoPsbt.js +657 -121
  19. package/dist/src/bitgo/UtxoTransaction.js +2 -2
  20. package/dist/src/bitgo/bitcoincash/address.js +2 -2
  21. package/dist/src/bitgo/index.d.ts +11 -0
  22. package/dist/src/bitgo/index.d.ts.map +1 -1
  23. package/dist/src/bitgo/index.js +6 -2
  24. package/dist/src/bitgo/legacysafe/index.d.ts +15 -0
  25. package/dist/src/bitgo/legacysafe/index.d.ts.map +1 -0
  26. package/dist/src/bitgo/legacysafe/index.js +61 -0
  27. package/dist/src/bitgo/litecoin/LitecoinPsbt.d.ts +10 -0
  28. package/dist/src/bitgo/litecoin/LitecoinPsbt.d.ts.map +1 -0
  29. package/dist/src/bitgo/litecoin/LitecoinPsbt.js +17 -0
  30. package/dist/src/bitgo/litecoin/LitecoinTransaction.d.ts +16 -0
  31. package/dist/src/bitgo/litecoin/LitecoinTransaction.d.ts.map +1 -0
  32. package/dist/src/bitgo/litecoin/LitecoinTransaction.js +46 -0
  33. package/dist/src/bitgo/litecoin/LitecoinTransactionBuilder.d.ts +10 -0
  34. package/dist/src/bitgo/litecoin/LitecoinTransactionBuilder.d.ts.map +1 -0
  35. package/dist/src/bitgo/litecoin/LitecoinTransactionBuilder.js +15 -0
  36. package/dist/src/bitgo/litecoin/index.d.ts +4 -0
  37. package/dist/src/bitgo/litecoin/index.d.ts.map +1 -0
  38. package/dist/src/bitgo/litecoin/index.js +16 -0
  39. package/dist/src/bitgo/outputScripts.d.ts +3 -1
  40. package/dist/src/bitgo/outputScripts.d.ts.map +1 -1
  41. package/dist/src/bitgo/outputScripts.js +20 -12
  42. package/dist/src/bitgo/parseInput.d.ts +49 -20
  43. package/dist/src/bitgo/parseInput.d.ts.map +1 -1
  44. package/dist/src/bitgo/parseInput.js +110 -26
  45. package/dist/src/bitgo/psbt/fromHalfSigned.d.ts.map +1 -1
  46. package/dist/src/bitgo/psbt/fromHalfSigned.js +9 -6
  47. package/dist/src/bitgo/psbt/scriptTypes.js +3 -3
  48. package/dist/src/bitgo/signature.d.ts +3 -3
  49. package/dist/src/bitgo/signature.d.ts.map +1 -1
  50. package/dist/src/bitgo/signature.js +48 -16
  51. package/dist/src/bitgo/transaction.d.ts +18 -3
  52. package/dist/src/bitgo/transaction.d.ts.map +1 -1
  53. package/dist/src/bitgo/transaction.js +28 -15
  54. package/dist/src/bitgo/types.d.ts +2 -0
  55. package/dist/src/bitgo/types.d.ts.map +1 -1
  56. package/dist/src/bitgo/types.js +1 -1
  57. package/dist/src/bitgo/wallet/Psbt.d.ts +104 -12
  58. package/dist/src/bitgo/wallet/Psbt.d.ts.map +1 -1
  59. package/dist/src/bitgo/wallet/Psbt.js +285 -70
  60. package/dist/src/bitgo/wallet/Unspent.d.ts +28 -0
  61. package/dist/src/bitgo/wallet/Unspent.d.ts.map +1 -1
  62. package/dist/src/bitgo/wallet/Unspent.js +172 -68
  63. package/dist/src/bitgo/wallet/WalletOutput.d.ts +17 -1
  64. package/dist/src/bitgo/wallet/WalletOutput.d.ts.map +1 -1
  65. package/dist/src/bitgo/wallet/WalletOutput.js +64 -23
  66. package/dist/src/bitgo/wallet/chains.d.ts +2 -2
  67. package/dist/src/bitgo/wallet/chains.d.ts.map +1 -1
  68. package/dist/src/bitgo/wallet/chains.js +1 -1
  69. package/dist/src/bitgo/zcash/ZcashPsbt.d.ts +0 -1
  70. package/dist/src/bitgo/zcash/ZcashPsbt.d.ts.map +1 -1
  71. package/dist/src/bitgo/zcash/ZcashPsbt.js +7 -16
  72. package/dist/src/bitgo/zcash/ZcashTransaction.js +2 -2
  73. package/dist/src/musig.d.ts +390 -0
  74. package/dist/src/musig.d.ts.map +1 -0
  75. package/dist/src/musig.js +447 -0
  76. package/dist/src/networks.d.ts +1 -2
  77. package/dist/src/networks.d.ts.map +1 -1
  78. package/dist/src/networks.js +22 -29
  79. package/dist/src/noble_ecc.d.ts +1 -1
  80. package/dist/src/noble_ecc.d.ts.map +1 -1
  81. package/dist/src/noble_ecc.js +11 -7
  82. package/dist/src/payments/p2tr.d.ts.map +1 -1
  83. package/dist/src/payments/p2tr.js +21 -19
  84. package/dist/src/payments/p2tr_ns.js +2 -3
  85. package/dist/src/taproot.d.ts +16 -0
  86. package/dist/src/taproot.d.ts.map +1 -1
  87. package/dist/src/taproot.js +45 -4
  88. package/dist/src/testutil/index.d.ts +2 -0
  89. package/dist/src/testutil/index.d.ts.map +1 -1
  90. package/dist/src/testutil/index.js +3 -1
  91. package/dist/src/testutil/keys.d.ts +3 -0
  92. package/dist/src/testutil/keys.d.ts.map +1 -1
  93. package/dist/src/testutil/keys.js +17 -2
  94. package/dist/src/testutil/mock.d.ts +1 -1
  95. package/dist/src/testutil/mock.d.ts.map +1 -1
  96. package/dist/src/testutil/mock.js +12 -4
  97. package/dist/src/testutil/psbt.d.ts +89 -0
  98. package/dist/src/testutil/psbt.d.ts.map +1 -0
  99. package/dist/src/testutil/psbt.js +150 -0
  100. package/dist/src/testutil/transaction.d.ts +70 -0
  101. package/dist/src/testutil/transaction.d.ts.map +1 -0
  102. package/dist/src/testutil/transaction.js +107 -0
  103. package/dist/src/transaction_builder.js +2 -2
  104. package/package.json +6 -7
@@ -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"]}