@btc-vision/transaction 1.0.85 → 1.0.86
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/.gitattributes +2 -2
- package/browser/_version.d.ts +1 -1
- package/browser/index.js +1 -1
- package/browser/keypair/Wallet.d.ts +3 -0
- package/browser/transaction/builders/TapUnwarpTransaction.d.ts +40 -40
- package/browser/transaction/builders/UnwarpTransaction.d.ts +34 -34
- package/browser/utxo/UTXOManager.d.ts +7 -7
- package/build/Utils.d.ts +0 -0
- package/build/Utils.js +1 -0
- package/build/_version.d.ts +1 -1
- package/build/_version.js +1 -1
- package/build/consensus/metadata/RoswsellConsensus.d.ts +2 -0
- package/build/consensus/metadata/RoswsellConsensus.js +4 -0
- package/build/contracts/ContractMetadataManager.d.ts +0 -0
- package/build/contracts/ContractMetadataManager.js +1 -0
- package/build/generators/OPNetAddressGenerator.d.ts +0 -0
- package/build/generators/OPNetAddressGenerator.js +1 -0
- package/build/generators/builders/UnwrapGenerator.d.ts +8 -0
- package/build/generators/builders/UnwrapGenerator.js +79 -0
- package/build/keypair/Wallet.d.ts +3 -0
- package/build/keypair/Wallet.js +8 -0
- package/build/keypair/interfaces/GeneratedWallet.d.ts +5 -0
- package/build/keypair/interfaces/GeneratedWallet.js +1 -0
- package/build/metadata/CommonContracts.d.ts +6 -0
- package/build/metadata/CommonContracts.js +5 -0
- package/build/metadata/ContractMetadataManager.d.ts +1 -0
- package/build/metadata/ContractMetadataManager.js +9 -0
- package/build/metadata/contracts/ContractBase.d.ts +9 -0
- package/build/metadata/contracts/ContractBase.js +13 -0
- package/build/metadata/contracts/ContractBaseMetadata.d.ts +9 -0
- package/build/metadata/contracts/ContractBaseMetadata.js +13 -0
- package/build/metadata/contracts/ContractMetadataManager.d.ts +0 -0
- package/build/metadata/contracts/ContractMetadataManager.js +1 -0
- package/build/network/NetworkConverter.d.ts +0 -0
- package/build/network/NetworkConverter.js +14 -0
- package/build/scripts/Regtest.d.ts +2 -0
- package/build/scripts/Regtest.js +15 -0
- package/build/scripts/test.d.ts +1 -0
- package/build/scripts/test.js +74 -0
- package/build/signer/Regtest.d.ts +2 -0
- package/build/signer/Regtest.js +15 -0
- package/build/tests/Regtest.d.ts +3 -0
- package/build/tests/Regtest.js +29 -0
- package/build/tests/adaptPSBT.d.ts +1 -0
- package/build/tests/adaptPSBT.js +44 -0
- package/build/tests/btc/send.d.ts +1 -0
- package/build/tests/btc/send.js +35 -0
- package/build/tests/btc/transfer.d.ts +1 -0
- package/build/tests/btc/transfer.js +35 -0
- package/build/tests/createPairReg.d.ts +1 -0
- package/build/tests/createPairReg.js +73 -0
- package/build/tests/deploy/deployMoto.d.ts +4 -0
- package/build/tests/deploy/deployMoto.js +89 -0
- package/build/tests/deploy/deployPool.d.ts +1 -0
- package/build/tests/deploy/deployPool.js +5 -0
- package/build/tests/deploy/deployStep1.d.ts +1 -0
- package/build/tests/deploy/deployStep1.js +5 -0
- package/build/tests/deploy/deployStep2.d.ts +1 -0
- package/build/tests/deploy/deployStep2.js +5 -0
- package/build/tests/deploy/deployStep3.d.ts +1 -0
- package/build/tests/deploy/deployStep3.js +5 -0
- package/build/tests/deploy.d.ts +1 -0
- package/build/tests/deploy.js +41 -0
- package/build/tests/deployMotoRegStep1.d.ts +1 -0
- package/build/tests/deployMotoRegStep1.js +85 -0
- package/build/tests/deployReg.d.ts +1 -0
- package/build/tests/deployReg.js +85 -0
- package/build/tests/factory/createPairReg.d.ts +1 -0
- package/build/tests/factory/createPairReg.js +13 -0
- package/build/tests/gen.d.ts +1 -0
- package/build/tests/gen.js +19 -0
- package/build/tests/interaction.d.ts +5 -0
- package/build/tests/interaction.js +62 -0
- package/build/tests/massWrapReg.d.ts +1 -0
- package/build/tests/massWrapReg.js +105 -0
- package/build/tests/mineReg.d.ts +1 -0
- package/build/tests/mineReg.js +19 -0
- package/build/tests/moto/airdropToken.d.ts +1 -0
- package/build/tests/moto/airdropToken.js +21 -0
- package/build/tests/moto/airdropTokens.d.ts +1 -0
- package/build/tests/moto/airdropTokens.js +60 -0
- package/build/tests/moto/allowance.d.ts +1 -0
- package/build/tests/moto/allowance.js +6 -0
- package/build/tests/moto/approve.d.ts +1 -0
- package/build/tests/moto/approve.js +10 -0
- package/build/tests/moto/approveWBTC.d.ts +1 -0
- package/build/tests/moto/approveWBTC.js +12 -0
- package/build/tests/moto/balanceOf.d.ts +1 -0
- package/build/tests/moto/balanceOf.js +12 -0
- package/build/tests/moto/transfer.d.ts +1 -0
- package/build/tests/moto/transfer.js +16 -0
- package/build/tests/motoswap/airdropToken.d.ts +11 -0
- package/build/tests/motoswap/airdropToken.js +36 -0
- package/build/tests/motoswap/deployMoto.d.ts +4 -0
- package/build/tests/motoswap/deployMoto.js +89 -0
- package/build/tests/motoswap/deployMotoRegStep1.d.ts +1 -0
- package/build/tests/motoswap/deployMotoRegStep1.js +91 -0
- package/build/tests/motoswap/deployMotoRegStep2.d.ts +1 -0
- package/build/tests/motoswap/deployMotoRegStep2.js +91 -0
- package/build/tests/motoswap/deployPool.d.ts +1 -0
- package/build/tests/motoswap/deployPool.js +5 -0
- package/build/tests/motoswap/deployStep1.d.ts +1 -0
- package/build/tests/motoswap/deployStep1.js +5 -0
- package/build/tests/motoswap/deployStep2.d.ts +1 -0
- package/build/tests/motoswap/deployStep2.js +5 -0
- package/build/tests/motoswap/deployStep3.d.ts +1 -0
- package/build/tests/motoswap/deployStep3.js +5 -0
- package/build/tests/motoswap/interaction.d.ts +3 -0
- package/build/tests/motoswap/interaction.js +63 -0
- package/build/tests/motoswap/routerAddLiquidity.d.ts +11 -0
- package/build/tests/motoswap/routerAddLiquidity.js +35 -0
- package/build/tests/motoswap-router/addLiquidity.d.ts +11 -0
- package/build/tests/motoswap-router/addLiquidity.js +36 -0
- package/build/tests/motoswap-router/deployMoto.d.ts +4 -0
- package/build/tests/motoswap-router/deployMoto.js +89 -0
- package/build/tests/motoswap-router/deployPool.d.ts +1 -0
- package/build/tests/motoswap-router/deployPool.js +5 -0
- package/build/tests/motoswap-router/deployStep1.d.ts +1 -0
- package/build/tests/motoswap-router/deployStep1.js +5 -0
- package/build/tests/motoswap-router/deployStep2.d.ts +1 -0
- package/build/tests/motoswap-router/deployStep2.js +5 -0
- package/build/tests/motoswap-router/deployStep3.d.ts +1 -0
- package/build/tests/motoswap-router/deployStep3.js +5 -0
- package/build/tests/motoswap-router/getAmountsOut.d.ts +5 -0
- package/build/tests/motoswap-router/getAmountsOut.js +34 -0
- package/build/tests/motoswap-router/routerAddLiquidity.d.ts +11 -0
- package/build/tests/motoswap-router/routerAddLiquidity.js +35 -0
- package/build/tests/motoswap-router/swap.d.ts +8 -0
- package/build/tests/motoswap-router/swap.js +24 -0
- package/build/tests/multisign.d.ts +1 -0
- package/build/tests/multisign.js +47 -0
- package/build/tests/multisign2.d.ts +1 -0
- package/build/tests/multisign2.js +27 -0
- package/build/tests/pool/DecodePoolAddress.d.ts +6 -0
- package/build/tests/pool/DecodePoolAddress.js +12 -0
- package/build/tests/pool/decodeReserves.d.ts +5 -0
- package/build/tests/pool/decodeReserves.js +13 -0
- package/build/tests/pool/reserves.d.ts +1 -0
- package/build/tests/pool/reserves.js +18 -0
- package/build/tests/shared/Utils.d.ts +2 -0
- package/build/tests/shared/Utils.js +14 -0
- package/build/tests/shared/interaction.d.ts +7 -0
- package/build/tests/shared/interaction.js +85 -0
- package/build/tests/shared/tokens.d.ts +6 -0
- package/build/tests/shared/tokens.js +5 -0
- package/build/tests/stakeReg.d.ts +1 -0
- package/build/tests/stakeReg.js +73 -0
- package/build/tests/stakedReg.d.ts +1 -0
- package/build/tests/stakedReg.js +28 -0
- package/build/tests/test.d.ts +1 -0
- package/build/tests/test.js +51 -0
- package/build/tests/test2.d.ts +1 -0
- package/build/tests/test2.js +73 -0
- package/build/tests/testReg.d.ts +1 -0
- package/build/tests/testReg.js +91 -0
- package/build/tests/tokens.d.ts +6 -0
- package/build/tests/tokens.js +5 -0
- package/build/tests/totalRewardReg.d.ts +1 -0
- package/build/tests/totalRewardReg.js +28 -0
- package/build/tests/transfer.d.ts +1 -0
- package/build/tests/transfer.js +74 -0
- package/build/tests/transferReg.d.ts +1 -0
- package/build/tests/transferReg.js +74 -0
- package/build/tests/unStakeReg.d.ts +1 -0
- package/build/tests/unStakeReg.js +72 -0
- package/build/tests/unwrapReg.d.ts +1 -0
- package/build/tests/unwrapReg.js +61 -0
- package/build/tests/unwrapReg2.d.ts +1 -0
- package/build/tests/unwrapReg2.js +56 -0
- package/build/tests/unwrapRegSegwit.d.ts +1 -0
- package/build/tests/unwrapRegSegwit.js +83 -0
- package/build/tests/wbtc/approve.d.ts +1 -0
- package/build/tests/wbtc/approve.js +6 -0
- package/build/tests/wbtc/approveWBTC.d.ts +1 -0
- package/build/tests/wbtc/approveWBTC.js +12 -0
- package/build/tests/wbtc/massWrapReg.d.ts +1 -0
- package/build/tests/wbtc/massWrapReg.js +105 -0
- package/build/tests/wbtc/transfer.d.ts +1 -0
- package/build/tests/wbtc/transfer.js +16 -0
- package/build/tests/wbtc/transferReg.d.ts +1 -0
- package/build/tests/wbtc/transferReg.js +16 -0
- package/build/tests/wbtc/unStakeReg.d.ts +1 -0
- package/build/tests/wbtc/unStakeReg.js +72 -0
- package/build/tests/wbtc/unwrapReg.d.ts +1 -0
- package/build/tests/wbtc/unwrapReg.js +60 -0
- package/build/tests/wbtc/unwrapRegSegwit.d.ts +1 -0
- package/build/tests/wbtc/unwrapRegSegwit.js +83 -0
- package/build/tests/wbtc/withdrawalRequestReg.d.ts +1 -0
- package/build/tests/wbtc/withdrawalRequestReg.js +71 -0
- package/build/tests/wbtc/wrapReg.d.ts +1 -0
- package/build/tests/wbtc/wrapReg.js +65 -0
- package/build/tests/wbtc/wrapTest.d.ts +1 -0
- package/build/tests/wbtc/wrapTest.js +66 -0
- package/build/tests/withdrawalRequestReg.d.ts +1 -0
- package/build/tests/withdrawalRequestReg.js +71 -0
- package/build/tests/wrap.d.ts +1 -0
- package/build/tests/wrap.js +65 -0
- package/build/tests/wrapReg.d.ts +1 -0
- package/build/tests/wrapReg.js +68 -0
- package/build/tests/wrapTest.d.ts +1 -0
- package/build/tests/wrapTest.js +66 -0
- package/build/tests/wrapTestg.d.ts +1 -0
- package/build/tests/wrapTestg.js +66 -0
- package/build/tests/writers/allowance.d.ts +3 -0
- package/build/tests/writers/allowance.js +10 -0
- package/build/tests/writers/approve.d.ts +4 -0
- package/build/tests/writers/approve.js +11 -0
- package/build/transaction/TransactionBuilder.d.ts +60 -0
- package/build/transaction/TransactionBuilder.js +244 -0
- package/build/transaction/browser/BrowserSigner.d.ts +11 -0
- package/build/transaction/browser/BrowserSigner.js +10 -0
- package/build/transaction/browser/extensions/Unisat.d.ts +54 -0
- package/build/transaction/browser/extensions/Unisat.js +11 -0
- package/build/transaction/builders/GenericTransaction.d.ts +11 -0
- package/build/transaction/builders/GenericTransaction.js +23 -0
- package/build/transaction/builders/TapUnwarpTransaction.d.ts +37 -0
- package/build/transaction/builders/TapUnwarpTransaction.js +201 -0
- package/build/transaction/builders/UnwarpSegwitTransaction.d.ts +34 -0
- package/build/transaction/builders/UnwarpSegwitTransaction.js +184 -0
- package/build/transaction/builders/UnwarpTransaction.d.ts +35 -0
- package/build/transaction/builders/UnwarpTransaction.js +199 -0
- package/build/transaction/interfaces/ITransactions.d.ts +32 -0
- package/build/transaction/interfaces/ITransactions.js +1 -0
- package/build/utxo/IUTXO.d.ts +0 -0
- package/build/utxo/IUTXO.js +1 -0
- package/build/utxo/OPNetUtils.d.ts +7 -0
- package/build/utxo/OPNetUtils.js +47 -0
- package/build/utxo/UTXOManager.d.ts +7 -0
- package/build/utxo/UTXOManager.js +47 -0
- package/build/wbtc/BroadcastResponse.d.ts +0 -0
- package/build/wbtc/BroadcastResponse.js +1 -0
- package/gulpfile.js +152 -152
- package/package.json +109 -109
- package/src/_version.ts +1 -1
- package/src/consensus/Consensus.ts +36 -36
- package/src/consensus/ConsensusConfig.ts +39 -39
- package/src/crypto/crypto-browser.js +75 -75
- package/src/generators/AddressGenerator.ts +24 -24
- package/src/generators/Features.ts +5 -5
- package/src/generators/Generator.ts +75 -75
- package/src/generators/builders/CalldataGenerator.ts +148 -148
- package/src/generators/builders/DeploymentGenerator.ts +66 -66
- package/src/index.ts +4 -4
- package/src/keypair/AddressVerificator.ts +40 -40
- package/src/keypair/EcKeyPair.ts +282 -282
- package/src/keypair/Wallet.ts +120 -97
- package/src/keypair/interfaces/IWallet.ts +19 -19
- package/src/metadata/ContractBaseMetadata.ts +23 -23
- package/src/metadata/contracts/wBTC.ts +60 -60
- package/src/metadata/tokens.ts +135 -135
- package/src/network/NetworkInformation.ts +7 -7
- package/src/transaction/TransactionFactory.ts +496 -496
- package/src/transaction/browser/BrowserSignerBase.ts +37 -37
- package/src/transaction/browser/Web3Provider.ts +46 -46
- package/src/transaction/browser/extensions/UnisatSigner.ts +218 -218
- package/src/transaction/browser/types/Unisat.ts +97 -97
- package/src/transaction/builders/FundingTransaction.ts +40 -40
- package/src/transaction/builders/InteractionTransaction.ts +38 -38
- package/src/transaction/builders/SharedInteractionTransaction.ts +368 -368
- package/src/transaction/builders/TransactionBuilder.ts +665 -665
- package/src/transaction/builders/UnwrapSegwitTransaction.ts +365 -365
- package/src/transaction/builders/UnwrapTransaction.ts +507 -507
- package/src/transaction/builders/WrapTransaction.ts +346 -346
- package/src/transaction/interfaces/ITransactionParameters.ts +59 -59
- package/src/transaction/interfaces/Tap.ts +26 -26
- package/src/transaction/psbt/PSBTTypes.ts +3 -3
- package/src/transaction/shared/TweakedTransaction.ts +539 -539
- package/src/utxo/OPNetLimitedProvider.ts +244 -244
- package/src/utxo/interfaces/BroadcastResponse.ts +10 -10
- package/src/utxo/interfaces/IUTXO.ts +29 -29
- package/src/verification/TapscriptVerificator.ts +89 -89
- package/src/wbtc/Generate.ts +40 -40
- package/src/wbtc/UnwrapGeneration.ts +13 -13
- package/src/wbtc/WrappedGenerationParameters.ts +33 -33
- package/webpack.config.js +78 -78
- /package/build/generators/builders/{MultiSignGenerator.d.ts → MultisignGenerator.d.ts} +0 -0
- /package/build/generators/builders/{MultiSignGenerator.js → MultisignGenerator.js} +0 -0
- /package/build/generators/{Features.d.ts → features.d.ts} +0 -0
- /package/build/generators/{Features.js → features.js} +0 -0
|
@@ -0,0 +1,34 @@
|
|
|
1
|
+
/// <reference types="node" />
|
|
2
|
+
import { Taptree } from 'bitcoinjs-lib/src/types.js';
|
|
3
|
+
import { TransactionType } from '../enums/TransactionType.js';
|
|
4
|
+
import { IUnwrapParameters } from '../interfaces/ITransactionParameters.js';
|
|
5
|
+
import { SharedInteractionTransaction } from './SharedInteractionTransaction.js';
|
|
6
|
+
import { Psbt } from 'bitcoinjs-lib';
|
|
7
|
+
import { VaultUTXOs } from '../processor/PsbtTransaction.js';
|
|
8
|
+
export declare class UnwrapTransaction extends SharedInteractionTransaction<TransactionType.WBTC_UNWRAP> {
|
|
9
|
+
static readonly MINIMUM_CONSOLIDATION_AMOUNT: bigint;
|
|
10
|
+
private static readonly UNWRAP_SELECTOR;
|
|
11
|
+
type: TransactionType.WBTC_UNWRAP;
|
|
12
|
+
readonly amount: bigint;
|
|
13
|
+
protected readonly compiledTargetScript: Buffer;
|
|
14
|
+
protected readonly scriptTree: Taptree;
|
|
15
|
+
protected sighashTypes: number[];
|
|
16
|
+
protected readonly contractSecret: Buffer;
|
|
17
|
+
protected readonly vaultUTXOs: VaultUTXOs[];
|
|
18
|
+
private readonly wbtc;
|
|
19
|
+
private readonly calculatedSignHash;
|
|
20
|
+
constructor(parameters: IUnwrapParameters);
|
|
21
|
+
static generateBurnCalldata(amount: bigint): Buffer;
|
|
22
|
+
signPSBT(): Psbt;
|
|
23
|
+
mergeVaults(input: VaultUTXOs[]): void;
|
|
24
|
+
protected internalBuildTransaction(transaction: Psbt): boolean;
|
|
25
|
+
protected generateMultiSignRedeemScript(publicKeys: string[], minimum: number): {
|
|
26
|
+
witnessUtxo: Buffer;
|
|
27
|
+
redeemScript: Buffer;
|
|
28
|
+
witnessScript: Buffer;
|
|
29
|
+
};
|
|
30
|
+
private addVaultUTXO;
|
|
31
|
+
private addVaultInputs;
|
|
32
|
+
private calculateOutputLeftAmountFromVaults;
|
|
33
|
+
private getVaultTotalOutputAmount;
|
|
34
|
+
}
|
|
@@ -0,0 +1,184 @@
|
|
|
1
|
+
import { TransactionType } from '../enums/TransactionType.js';
|
|
2
|
+
import { SharedInteractionTransaction } from './SharedInteractionTransaction.js';
|
|
3
|
+
import { TransactionBuilder } from './TransactionBuilder.js';
|
|
4
|
+
import { ABICoder, BinaryWriter } from '@btc-vision/bsi-binary';
|
|
5
|
+
import { wBTC } from '../../metadata/contracts/wBTC.js';
|
|
6
|
+
import { payments } from 'bitcoinjs-lib';
|
|
7
|
+
import { EcKeyPair } from '../../keypair/EcKeyPair.js';
|
|
8
|
+
import { PsbtTransaction } from '../processor/PsbtTransaction.js';
|
|
9
|
+
const abiCoder = new ABICoder();
|
|
10
|
+
export class UnwrapTransaction extends SharedInteractionTransaction {
|
|
11
|
+
static MINIMUM_CONSOLIDATION_AMOUNT = 200000n;
|
|
12
|
+
static UNWRAP_SELECTOR = Number('0x' + abiCoder.encodeSelector('burn'));
|
|
13
|
+
type = TransactionType.WBTC_UNWRAP;
|
|
14
|
+
amount;
|
|
15
|
+
compiledTargetScript;
|
|
16
|
+
scriptTree;
|
|
17
|
+
sighashTypes = [];
|
|
18
|
+
contractSecret;
|
|
19
|
+
vaultUTXOs;
|
|
20
|
+
wbtc;
|
|
21
|
+
calculatedSignHash = PsbtTransaction.calculateSignHash(this.sighashTypes);
|
|
22
|
+
constructor(parameters) {
|
|
23
|
+
if (parameters.amount < TransactionBuilder.MINIMUM_DUST) {
|
|
24
|
+
throw new Error('Amount is below dust limit');
|
|
25
|
+
}
|
|
26
|
+
parameters.disableAutoRefund = true;
|
|
27
|
+
parameters.calldata = UnwrapTransaction.generateBurnCalldata(parameters.amount);
|
|
28
|
+
super(parameters);
|
|
29
|
+
this.wbtc = new wBTC(parameters.network);
|
|
30
|
+
this.to = this.wbtc.getAddress();
|
|
31
|
+
this.vaultUTXOs = parameters.unwrapUTXOs;
|
|
32
|
+
this.amount = parameters.amount;
|
|
33
|
+
this.contractSecret = this.generateSecret();
|
|
34
|
+
this.compiledTargetScript = this.calldataGenerator.compile(this.calldata, this.contractSecret);
|
|
35
|
+
this.scriptTree = this.getScriptTree();
|
|
36
|
+
this.internalInit();
|
|
37
|
+
}
|
|
38
|
+
static generateBurnCalldata(amount) {
|
|
39
|
+
if (!amount)
|
|
40
|
+
throw new Error('Amount is required');
|
|
41
|
+
const bufWriter = new BinaryWriter();
|
|
42
|
+
bufWriter.writeSelector(UnwrapTransaction.UNWRAP_SELECTOR);
|
|
43
|
+
bufWriter.writeU256(amount);
|
|
44
|
+
return Buffer.from(bufWriter.getBuffer());
|
|
45
|
+
}
|
|
46
|
+
signPSBT() {
|
|
47
|
+
if (this.to && !EcKeyPair.verifyContractAddress(this.to, this.network)) {
|
|
48
|
+
throw new Error('Invalid contract address. The contract address must be a taproot address.');
|
|
49
|
+
}
|
|
50
|
+
if (!this.vaultUTXOs.length) {
|
|
51
|
+
throw new Error('No vault UTXOs provided');
|
|
52
|
+
}
|
|
53
|
+
if (this.signed)
|
|
54
|
+
throw new Error('Transaction is already signed');
|
|
55
|
+
this.signed = true;
|
|
56
|
+
this.buildTransaction();
|
|
57
|
+
this.ignoreSignatureError();
|
|
58
|
+
this.mergeVaults(this.vaultUTXOs);
|
|
59
|
+
const builtTx = this.internalBuildTransaction(this.transaction);
|
|
60
|
+
if (builtTx) {
|
|
61
|
+
return this.transaction;
|
|
62
|
+
}
|
|
63
|
+
throw new Error('Could not sign transaction');
|
|
64
|
+
}
|
|
65
|
+
mergeVaults(input) {
|
|
66
|
+
const firstVault = input[0];
|
|
67
|
+
if (!firstVault) {
|
|
68
|
+
throw new Error('No vaults provided');
|
|
69
|
+
}
|
|
70
|
+
const total = this.getVaultTotalOutputAmount(input);
|
|
71
|
+
if (total < this.amount) {
|
|
72
|
+
throw new Error(`Total vault amount (${total} sat) is less than the amount to unwrap (${this.amount} sat)`);
|
|
73
|
+
}
|
|
74
|
+
const outputLeftAmount = this.calculateOutputLeftAmountFromVaults(input);
|
|
75
|
+
if (outputLeftAmount < UnwrapTransaction.MINIMUM_CONSOLIDATION_AMOUNT) {
|
|
76
|
+
throw new Error(`Output left amount is below minimum consolidation (${UnwrapTransaction.MINIMUM_CONSOLIDATION_AMOUNT} sat) amount ${outputLeftAmount} for vault ${firstVault.vault}`);
|
|
77
|
+
}
|
|
78
|
+
this.addOutput({
|
|
79
|
+
address: firstVault.vault,
|
|
80
|
+
value: Number(outputLeftAmount),
|
|
81
|
+
});
|
|
82
|
+
this.addOutput({
|
|
83
|
+
address: this.from,
|
|
84
|
+
value: Number(this.amount),
|
|
85
|
+
});
|
|
86
|
+
for (const vault of input) {
|
|
87
|
+
this.addVaultInputs(vault);
|
|
88
|
+
}
|
|
89
|
+
}
|
|
90
|
+
internalBuildTransaction(transaction) {
|
|
91
|
+
if (transaction.data.inputs.length === 0) {
|
|
92
|
+
const inputs = this.getInputs();
|
|
93
|
+
const outputs = this.getOutputs();
|
|
94
|
+
transaction.setMaximumFeeRate(this._maximumFeeRate);
|
|
95
|
+
transaction.addInputs(inputs);
|
|
96
|
+
for (let i = 0; i < this.updateInputs.length; i++) {
|
|
97
|
+
transaction.updateInput(i, this.updateInputs[i]);
|
|
98
|
+
}
|
|
99
|
+
transaction.addOutputs(outputs);
|
|
100
|
+
}
|
|
101
|
+
try {
|
|
102
|
+
this.signInputs(transaction);
|
|
103
|
+
if (this.finalized) {
|
|
104
|
+
this.transactionFee = BigInt(transaction.getFee());
|
|
105
|
+
}
|
|
106
|
+
return true;
|
|
107
|
+
}
|
|
108
|
+
catch (e) {
|
|
109
|
+
const err = e;
|
|
110
|
+
this.error(`[internalBuildTransaction] Something went wrong while getting building the transaction: ${err.stack}`);
|
|
111
|
+
}
|
|
112
|
+
return false;
|
|
113
|
+
}
|
|
114
|
+
generateMultiSignRedeemScript(publicKeys, minimum) {
|
|
115
|
+
const p2ms = payments.p2ms({
|
|
116
|
+
m: minimum,
|
|
117
|
+
pubkeys: publicKeys.map((key) => Buffer.from(key, 'base64')),
|
|
118
|
+
network: this.network,
|
|
119
|
+
});
|
|
120
|
+
const p2wsh = payments.p2wsh({
|
|
121
|
+
redeem: p2ms,
|
|
122
|
+
network: this.network,
|
|
123
|
+
});
|
|
124
|
+
const witnessUtxo = p2wsh.output;
|
|
125
|
+
const redeemScript = p2wsh.redeem?.output;
|
|
126
|
+
const witnessScript = p2ms.output;
|
|
127
|
+
if (!witnessUtxo || !redeemScript || !witnessScript) {
|
|
128
|
+
throw new Error('Failed to generate redeem script');
|
|
129
|
+
}
|
|
130
|
+
return {
|
|
131
|
+
witnessUtxo,
|
|
132
|
+
redeemScript,
|
|
133
|
+
witnessScript,
|
|
134
|
+
};
|
|
135
|
+
}
|
|
136
|
+
addVaultUTXO(utxo, witness) {
|
|
137
|
+
const input = {
|
|
138
|
+
hash: utxo.hash,
|
|
139
|
+
index: utxo.outputIndex,
|
|
140
|
+
witnessUtxo: {
|
|
141
|
+
script: Buffer.from(utxo.output, 'base64'),
|
|
142
|
+
value: Number(utxo.value),
|
|
143
|
+
},
|
|
144
|
+
witnessScript: witness.witnessScript,
|
|
145
|
+
sequence: this.sequence,
|
|
146
|
+
};
|
|
147
|
+
if (this.calculatedSignHash) {
|
|
148
|
+
input.sighashType = this.calculatedSignHash;
|
|
149
|
+
}
|
|
150
|
+
this.addInput(input);
|
|
151
|
+
}
|
|
152
|
+
addVaultInputs(vault, firstSigner = this.signer) {
|
|
153
|
+
const p2wshOutput = this.generateMultiSignRedeemScript(vault.publicKeys, vault.minimum);
|
|
154
|
+
for (const utxo of vault.utxos) {
|
|
155
|
+
const inputIndex = this.transaction.inputCount;
|
|
156
|
+
this.addVaultUTXO(utxo, p2wshOutput);
|
|
157
|
+
if (firstSigner) {
|
|
158
|
+
this.log(`Signing input ${inputIndex} with ${firstSigner.publicKey.toString('hex')}`);
|
|
159
|
+
try {
|
|
160
|
+
this.signInput(this.transaction, this.transaction.data.inputs[inputIndex], inputIndex, this.signer);
|
|
161
|
+
this.log(`Signed input ${inputIndex} with ${firstSigner.publicKey.toString('hex')}`);
|
|
162
|
+
}
|
|
163
|
+
catch (e) {
|
|
164
|
+
if (!this.ignoreSignatureErrors) {
|
|
165
|
+
this.warn(`Failed to sign input ${inputIndex} with ${firstSigner.publicKey.toString('hex')} ${e.message}`);
|
|
166
|
+
}
|
|
167
|
+
}
|
|
168
|
+
}
|
|
169
|
+
}
|
|
170
|
+
}
|
|
171
|
+
calculateOutputLeftAmountFromVaults(vaults) {
|
|
172
|
+
const total = this.getVaultTotalOutputAmount(vaults);
|
|
173
|
+
return total - this.amount;
|
|
174
|
+
}
|
|
175
|
+
getVaultTotalOutputAmount(vaults) {
|
|
176
|
+
let total = BigInt(0);
|
|
177
|
+
for (const vault of vaults) {
|
|
178
|
+
for (const utxo of vault.utxos) {
|
|
179
|
+
total += BigInt(utxo.value);
|
|
180
|
+
}
|
|
181
|
+
}
|
|
182
|
+
return total;
|
|
183
|
+
}
|
|
184
|
+
}
|
|
@@ -0,0 +1,35 @@
|
|
|
1
|
+
/// <reference types="node" />
|
|
2
|
+
import { Taptree } from 'bitcoinjs-lib/src/types.js';
|
|
3
|
+
import { TransactionType } from '../enums/TransactionType.js';
|
|
4
|
+
import { IUnwrapParameters } from '../interfaces/ITransactionParameters.js';
|
|
5
|
+
import { SharedInteractionTransaction } from './SharedInteractionTransaction.js';
|
|
6
|
+
import { Psbt } from 'bitcoinjs-lib';
|
|
7
|
+
import { VaultUTXOs } from '../processor/PsbtTransaction.js';
|
|
8
|
+
export declare class UnwrapTransaction extends SharedInteractionTransaction<TransactionType.WBTC_UNWRAP> {
|
|
9
|
+
static readonly MINIMUM_CONSOLIDATION_AMOUNT: bigint;
|
|
10
|
+
private static readonly UNWRAP_SELECTOR;
|
|
11
|
+
type: TransactionType.WBTC_UNWRAP;
|
|
12
|
+
readonly amount: bigint;
|
|
13
|
+
protected readonly compiledTargetScript: Buffer;
|
|
14
|
+
protected readonly scriptTree: Taptree;
|
|
15
|
+
protected sighashTypes: number[];
|
|
16
|
+
protected readonly contractSecret: Buffer;
|
|
17
|
+
protected readonly vaultUTXOs: VaultUTXOs[];
|
|
18
|
+
private readonly wbtc;
|
|
19
|
+
private readonly calculatedSignHash;
|
|
20
|
+
constructor(parameters: IUnwrapParameters);
|
|
21
|
+
static generateBurnCalldata(amount: bigint): Buffer;
|
|
22
|
+
signPSBT(): Psbt;
|
|
23
|
+
mergeVaults(input: VaultUTXOs[]): void;
|
|
24
|
+
estimateVaultFees(feeRate: bigint, numInputs: bigint, numOutputs: bigint, numSignatures: bigint, numPubkeys: bigint): bigint;
|
|
25
|
+
protected internalBuildTransaction(transaction: Psbt): boolean;
|
|
26
|
+
protected generateMultiSignRedeemScript(publicKeys: string[], minimum: number): {
|
|
27
|
+
witnessUtxo: Buffer;
|
|
28
|
+
redeemScript: Buffer;
|
|
29
|
+
witnessScript: Buffer;
|
|
30
|
+
};
|
|
31
|
+
private addVaultUTXO;
|
|
32
|
+
private addVaultInputs;
|
|
33
|
+
private calculateOutputLeftAmountFromVaults;
|
|
34
|
+
private getVaultTotalOutputAmount;
|
|
35
|
+
}
|
|
@@ -0,0 +1,199 @@
|
|
|
1
|
+
import { TransactionType } from '../enums/TransactionType.js';
|
|
2
|
+
import { SharedInteractionTransaction } from './SharedInteractionTransaction.js';
|
|
3
|
+
import { TransactionBuilder } from './TransactionBuilder.js';
|
|
4
|
+
import { ABICoder, BinaryWriter } from '@btc-vision/bsi-binary';
|
|
5
|
+
import { wBTC } from '../../metadata/contracts/wBTC.js';
|
|
6
|
+
import { payments } from 'bitcoinjs-lib';
|
|
7
|
+
import { EcKeyPair } from '../../keypair/EcKeyPair.js';
|
|
8
|
+
import { PsbtTransaction } from '../processor/PsbtTransaction.js';
|
|
9
|
+
const abiCoder = new ABICoder();
|
|
10
|
+
export class UnwrapTransaction extends SharedInteractionTransaction {
|
|
11
|
+
static MINIMUM_CONSOLIDATION_AMOUNT = 200000n;
|
|
12
|
+
static UNWRAP_SELECTOR = Number('0x' + abiCoder.encodeSelector('burn'));
|
|
13
|
+
type = TransactionType.WBTC_UNWRAP;
|
|
14
|
+
amount;
|
|
15
|
+
compiledTargetScript;
|
|
16
|
+
scriptTree;
|
|
17
|
+
sighashTypes = [];
|
|
18
|
+
contractSecret;
|
|
19
|
+
vaultUTXOs;
|
|
20
|
+
wbtc;
|
|
21
|
+
calculatedSignHash = PsbtTransaction.calculateSignHash(this.sighashTypes);
|
|
22
|
+
constructor(parameters) {
|
|
23
|
+
if (parameters.amount < TransactionBuilder.MINIMUM_DUST) {
|
|
24
|
+
throw new Error('Amount is below dust limit');
|
|
25
|
+
}
|
|
26
|
+
parameters.disableAutoRefund = true;
|
|
27
|
+
parameters.calldata = UnwrapTransaction.generateBurnCalldata(parameters.amount);
|
|
28
|
+
super(parameters);
|
|
29
|
+
this.wbtc = new wBTC(parameters.network);
|
|
30
|
+
this.to = this.wbtc.getAddress();
|
|
31
|
+
this.vaultUTXOs = parameters.unwrapUTXOs;
|
|
32
|
+
this.amount = parameters.amount;
|
|
33
|
+
this.contractSecret = this.generateSecret();
|
|
34
|
+
this.compiledTargetScript = this.calldataGenerator.compile(this.calldata, this.contractSecret);
|
|
35
|
+
this.scriptTree = this.getScriptTree();
|
|
36
|
+
this.internalInit();
|
|
37
|
+
}
|
|
38
|
+
static generateBurnCalldata(amount) {
|
|
39
|
+
if (!amount)
|
|
40
|
+
throw new Error('Amount is required');
|
|
41
|
+
const bufWriter = new BinaryWriter();
|
|
42
|
+
bufWriter.writeSelector(UnwrapTransaction.UNWRAP_SELECTOR);
|
|
43
|
+
bufWriter.writeU256(amount);
|
|
44
|
+
return Buffer.from(bufWriter.getBuffer());
|
|
45
|
+
}
|
|
46
|
+
signPSBT() {
|
|
47
|
+
if (this.to && !EcKeyPair.verifyContractAddress(this.to, this.network)) {
|
|
48
|
+
throw new Error('Invalid contract address. The contract address must be a taproot address.');
|
|
49
|
+
}
|
|
50
|
+
if (!this.vaultUTXOs.length) {
|
|
51
|
+
throw new Error('No vault UTXOs provided');
|
|
52
|
+
}
|
|
53
|
+
if (this.signed)
|
|
54
|
+
throw new Error('Transaction is already signed');
|
|
55
|
+
this.signed = true;
|
|
56
|
+
this.buildTransaction();
|
|
57
|
+
this.ignoreSignatureError();
|
|
58
|
+
this.mergeVaults(this.vaultUTXOs);
|
|
59
|
+
const builtTx = this.internalBuildTransaction(this.transaction);
|
|
60
|
+
if (builtTx) {
|
|
61
|
+
return this.transaction;
|
|
62
|
+
}
|
|
63
|
+
throw new Error('Could not sign transaction');
|
|
64
|
+
}
|
|
65
|
+
mergeVaults(input) {
|
|
66
|
+
const firstVault = input[0];
|
|
67
|
+
if (!firstVault) {
|
|
68
|
+
throw new Error('No vaults provided');
|
|
69
|
+
}
|
|
70
|
+
const total = this.getVaultTotalOutputAmount(input);
|
|
71
|
+
if (total < this.amount) {
|
|
72
|
+
throw new Error(`Total vault amount (${total} sat) is less than the amount to unwrap (${this.amount} sat)`);
|
|
73
|
+
}
|
|
74
|
+
const outputLeftAmount = this.calculateOutputLeftAmountFromVaults(input);
|
|
75
|
+
console.log('outputLeftAmount', outputLeftAmount, this.amount);
|
|
76
|
+
if (outputLeftAmount < UnwrapTransaction.MINIMUM_CONSOLIDATION_AMOUNT) {
|
|
77
|
+
throw new Error(`Output left amount is below minimum consolidation (${UnwrapTransaction.MINIMUM_CONSOLIDATION_AMOUNT} sat) amount ${outputLeftAmount} for vault ${firstVault.vault}`);
|
|
78
|
+
}
|
|
79
|
+
this.addOutput({
|
|
80
|
+
address: firstVault.vault,
|
|
81
|
+
value: Number(outputLeftAmount),
|
|
82
|
+
});
|
|
83
|
+
this.addOutput({
|
|
84
|
+
address: this.from,
|
|
85
|
+
value: Number(this.amount),
|
|
86
|
+
});
|
|
87
|
+
for (const vault of input) {
|
|
88
|
+
this.addVaultInputs(vault);
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
estimateVaultFees(feeRate, numInputs, numOutputs, numSignatures, numPubkeys) {
|
|
92
|
+
const txHeaderSize = 10n;
|
|
93
|
+
const inputBaseSize = 41n;
|
|
94
|
+
const outputSize = 68n;
|
|
95
|
+
const signatureSize = 144n;
|
|
96
|
+
const pubkeySize = 34n;
|
|
97
|
+
const baseTxSize = txHeaderSize + inputBaseSize * numInputs + outputSize * numOutputs;
|
|
98
|
+
const redeemScriptSize = 1n + numPubkeys * (1n + pubkeySize) + 1n + numSignatures;
|
|
99
|
+
const witnessSize = numSignatures * signatureSize + numPubkeys * pubkeySize + redeemScriptSize;
|
|
100
|
+
const weight = baseTxSize * 3n + (baseTxSize + witnessSize);
|
|
101
|
+
const vSize = weight / 4n;
|
|
102
|
+
return vSize * feeRate;
|
|
103
|
+
}
|
|
104
|
+
internalBuildTransaction(transaction) {
|
|
105
|
+
if (transaction.data.inputs.length === 0) {
|
|
106
|
+
const inputs = this.getInputs();
|
|
107
|
+
const outputs = this.getOutputs();
|
|
108
|
+
transaction.setMaximumFeeRate(this._maximumFeeRate);
|
|
109
|
+
transaction.addInputs(inputs);
|
|
110
|
+
for (let i = 0; i < this.updateInputs.length; i++) {
|
|
111
|
+
transaction.updateInput(i, this.updateInputs[i]);
|
|
112
|
+
}
|
|
113
|
+
transaction.addOutputs(outputs);
|
|
114
|
+
}
|
|
115
|
+
try {
|
|
116
|
+
this.signInputs(transaction);
|
|
117
|
+
if (this.finalized) {
|
|
118
|
+
this.transactionFee = BigInt(transaction.getFee());
|
|
119
|
+
}
|
|
120
|
+
return true;
|
|
121
|
+
}
|
|
122
|
+
catch (e) {
|
|
123
|
+
const err = e;
|
|
124
|
+
this.error(`[internalBuildTransaction] Something went wrong while getting building the transaction: ${err.stack}`);
|
|
125
|
+
}
|
|
126
|
+
return false;
|
|
127
|
+
}
|
|
128
|
+
generateMultiSignRedeemScript(publicKeys, minimum) {
|
|
129
|
+
const p2ms = payments.p2ms({
|
|
130
|
+
m: minimum,
|
|
131
|
+
pubkeys: publicKeys.map((key) => Buffer.from(key, 'base64')),
|
|
132
|
+
network: this.network,
|
|
133
|
+
});
|
|
134
|
+
const p2wsh = payments.p2wsh({
|
|
135
|
+
redeem: p2ms,
|
|
136
|
+
network: this.network,
|
|
137
|
+
});
|
|
138
|
+
const witnessUtxo = p2wsh.output;
|
|
139
|
+
const redeemScript = p2wsh.redeem?.output;
|
|
140
|
+
const witnessScript = p2ms.output;
|
|
141
|
+
if (!witnessUtxo || !redeemScript || !witnessScript) {
|
|
142
|
+
throw new Error('Failed to generate redeem script');
|
|
143
|
+
}
|
|
144
|
+
return {
|
|
145
|
+
witnessUtxo,
|
|
146
|
+
redeemScript,
|
|
147
|
+
witnessScript,
|
|
148
|
+
};
|
|
149
|
+
}
|
|
150
|
+
addVaultUTXO(utxo, witness) {
|
|
151
|
+
console.log(Number(utxo.value), utxo.hash, utxo.outputIndex);
|
|
152
|
+
const input = {
|
|
153
|
+
hash: utxo.hash,
|
|
154
|
+
index: utxo.outputIndex,
|
|
155
|
+
witnessUtxo: {
|
|
156
|
+
script: Buffer.from(utxo.output, 'base64'),
|
|
157
|
+
value: Number(utxo.value),
|
|
158
|
+
},
|
|
159
|
+
witnessScript: witness.witnessScript,
|
|
160
|
+
sequence: this.sequence,
|
|
161
|
+
};
|
|
162
|
+
if (this.calculatedSignHash) {
|
|
163
|
+
input.sighashType = this.calculatedSignHash;
|
|
164
|
+
}
|
|
165
|
+
this.addInput(input);
|
|
166
|
+
}
|
|
167
|
+
addVaultInputs(vault, firstSigner = this.signer) {
|
|
168
|
+
const p2wshOutput = this.generateMultiSignRedeemScript(vault.publicKeys, vault.minimum);
|
|
169
|
+
for (const utxo of vault.utxos) {
|
|
170
|
+
const inputIndex = this.transaction.inputCount;
|
|
171
|
+
this.addVaultUTXO(utxo, p2wshOutput);
|
|
172
|
+
if (firstSigner) {
|
|
173
|
+
this.log(`Signing input ${inputIndex} with ${firstSigner.publicKey.toString('hex')}`);
|
|
174
|
+
try {
|
|
175
|
+
this.signInput(this.transaction, this.transaction.data.inputs[inputIndex], inputIndex, this.signer);
|
|
176
|
+
this.log(`Signed input ${inputIndex} with ${firstSigner.publicKey.toString('hex')}`);
|
|
177
|
+
}
|
|
178
|
+
catch (e) {
|
|
179
|
+
if (!this.ignoreSignatureErrors) {
|
|
180
|
+
this.warn(`Failed to sign input ${inputIndex} with ${firstSigner.publicKey.toString('hex')} ${e.message}`);
|
|
181
|
+
}
|
|
182
|
+
}
|
|
183
|
+
}
|
|
184
|
+
}
|
|
185
|
+
}
|
|
186
|
+
calculateOutputLeftAmountFromVaults(vaults) {
|
|
187
|
+
const total = this.getVaultTotalOutputAmount(vaults);
|
|
188
|
+
return total - this.amount;
|
|
189
|
+
}
|
|
190
|
+
getVaultTotalOutputAmount(vaults) {
|
|
191
|
+
let total = BigInt(0);
|
|
192
|
+
for (const vault of vaults) {
|
|
193
|
+
for (const utxo of vault.utxos) {
|
|
194
|
+
total += BigInt(utxo.value);
|
|
195
|
+
}
|
|
196
|
+
}
|
|
197
|
+
return total;
|
|
198
|
+
}
|
|
199
|
+
}
|
|
@@ -0,0 +1,32 @@
|
|
|
1
|
+
/// <reference types="node" />
|
|
2
|
+
import { Signer } from 'bitcoinjs-lib';
|
|
3
|
+
import { UTXO } from '../../utxo/interfaces/IUTXO.js';
|
|
4
|
+
import { ECPairInterface } from 'ecpair';
|
|
5
|
+
import { Network } from 'bitcoinjs-lib/src/networks.js';
|
|
6
|
+
export interface ITransactionParameters {
|
|
7
|
+
readonly from: string;
|
|
8
|
+
readonly utxos: UTXO[];
|
|
9
|
+
readonly signer: ECPairInterface;
|
|
10
|
+
readonly network: Network;
|
|
11
|
+
readonly feeRate: number;
|
|
12
|
+
readonly priorityFee: bigint;
|
|
13
|
+
}
|
|
14
|
+
export interface SewgitTransactionParameters extends ITransactionParameters {
|
|
15
|
+
}
|
|
16
|
+
export interface ITransactionDataContractInteraction extends ITransactionParameters {
|
|
17
|
+
readonly contractSecret: Buffer;
|
|
18
|
+
readonly calldata: Buffer;
|
|
19
|
+
readonly customSigner: Signer;
|
|
20
|
+
readonly pubKeys?: Buffer[];
|
|
21
|
+
readonly minimumSignatures?: number;
|
|
22
|
+
}
|
|
23
|
+
export interface ITransactionDataContractInteractionWrap extends ITransactionDataContractInteraction {
|
|
24
|
+
readonly amount: bigint;
|
|
25
|
+
readonly minimumSignatures: number;
|
|
26
|
+
readonly pubKeys: Buffer[];
|
|
27
|
+
}
|
|
28
|
+
export interface ITransactionDataContractDeployment extends ITransactionParameters {
|
|
29
|
+
readonly bytecode: Buffer;
|
|
30
|
+
readonly salt: Buffer;
|
|
31
|
+
readonly customSigner: Signer;
|
|
32
|
+
}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
File without changes
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
"use strict";
|
|
@@ -0,0 +1,47 @@
|
|
|
1
|
+
export class UTXOManager {
|
|
2
|
+
opnetAPIUrl;
|
|
3
|
+
utxoPath = 'address/utxos';
|
|
4
|
+
constructor(opnetAPIUrl) {
|
|
5
|
+
this.opnetAPIUrl = opnetAPIUrl;
|
|
6
|
+
}
|
|
7
|
+
async fetchUTXO(settings) {
|
|
8
|
+
const params = {
|
|
9
|
+
method: 'GET',
|
|
10
|
+
headers: {
|
|
11
|
+
'Content-Type': 'application/json',
|
|
12
|
+
},
|
|
13
|
+
};
|
|
14
|
+
const url = `${this.opnetAPIUrl}/api/v1/${this.utxoPath}?address=${settings.address}`;
|
|
15
|
+
const resp = await fetch(url, params);
|
|
16
|
+
if (!resp.ok) {
|
|
17
|
+
throw new Error(`Failed to fetch UTXO data: ${resp.statusText}`);
|
|
18
|
+
}
|
|
19
|
+
const fetchedData = await resp.json();
|
|
20
|
+
if (fetchedData.length === 0) {
|
|
21
|
+
throw new Error('No UTXO found');
|
|
22
|
+
}
|
|
23
|
+
const meetCriteria = fetchedData.filter((utxo) => {
|
|
24
|
+
return BigInt(utxo.value) >= settings.minAmount;
|
|
25
|
+
});
|
|
26
|
+
if (meetCriteria.length === 0) {
|
|
27
|
+
throw new Error('No UTXO found (minAmount)');
|
|
28
|
+
}
|
|
29
|
+
let finalUTXOs = [];
|
|
30
|
+
let currentAmount = 0n;
|
|
31
|
+
const amountRequested = settings.requestedAmount;
|
|
32
|
+
for (const utxo of meetCriteria) {
|
|
33
|
+
const utxoValue = BigInt(utxo.value);
|
|
34
|
+
currentAmount += utxoValue;
|
|
35
|
+
finalUTXOs.push({
|
|
36
|
+
transactionId: utxo.transactionId,
|
|
37
|
+
outputIndex: utxo.outputIndex,
|
|
38
|
+
value: utxoValue,
|
|
39
|
+
scriptPubKey: utxo.scriptPubKey,
|
|
40
|
+
});
|
|
41
|
+
if (currentAmount + utxoValue > amountRequested) {
|
|
42
|
+
break;
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
return finalUTXOs;
|
|
46
|
+
}
|
|
47
|
+
}
|
|
@@ -0,0 +1,47 @@
|
|
|
1
|
+
export class UTXOManager {
|
|
2
|
+
opnetAPIUrl;
|
|
3
|
+
utxoPath = 'address/utxos';
|
|
4
|
+
constructor(opnetAPIUrl) {
|
|
5
|
+
this.opnetAPIUrl = opnetAPIUrl;
|
|
6
|
+
}
|
|
7
|
+
async fetchUTXO(settings) {
|
|
8
|
+
const params = {
|
|
9
|
+
method: 'GET',
|
|
10
|
+
headers: {
|
|
11
|
+
'Content-Type': 'application/json',
|
|
12
|
+
},
|
|
13
|
+
};
|
|
14
|
+
const url = `${this.opnetAPIUrl}/api/v1/${this.utxoPath}?address=${settings.address}`;
|
|
15
|
+
const resp = await fetch(url, params);
|
|
16
|
+
if (!resp.ok) {
|
|
17
|
+
throw new Error(`Failed to fetch UTXO data: ${resp.statusText}`);
|
|
18
|
+
}
|
|
19
|
+
const fetchedData = await resp.json();
|
|
20
|
+
if (fetchedData.length === 0) {
|
|
21
|
+
throw new Error('No UTXO found');
|
|
22
|
+
}
|
|
23
|
+
const meetCriteria = fetchedData.filter((utxo) => {
|
|
24
|
+
return BigInt(utxo.value) >= settings.minAmount;
|
|
25
|
+
});
|
|
26
|
+
if (meetCriteria.length === 0) {
|
|
27
|
+
throw new Error('No UTXO found (minAmount)');
|
|
28
|
+
}
|
|
29
|
+
let finalUTXOs = [];
|
|
30
|
+
let currentAmount = 0n;
|
|
31
|
+
const amountRequested = settings.requestedAmount;
|
|
32
|
+
for (const utxo of meetCriteria) {
|
|
33
|
+
const utxoValue = BigInt(utxo.value);
|
|
34
|
+
currentAmount += utxoValue;
|
|
35
|
+
finalUTXOs.push({
|
|
36
|
+
transactionId: utxo.transactionId,
|
|
37
|
+
outputIndex: utxo.outputIndex,
|
|
38
|
+
value: utxoValue,
|
|
39
|
+
scriptPubKey: utxo.scriptPubKey,
|
|
40
|
+
});
|
|
41
|
+
if (currentAmount + utxoValue > amountRequested) {
|
|
42
|
+
break;
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
return finalUTXOs;
|
|
46
|
+
}
|
|
47
|
+
}
|
|
File without changes
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
"use strict";
|