mirror of
https://github.com/cake-tech/cake_wallet.git
synced 2025-01-08 20:09:24 +00:00
update transaction build function
This commit is contained in:
parent
e97a3142fd
commit
282e34d0ea
5 changed files with 138 additions and 42 deletions
|
@ -4,6 +4,7 @@ import 'package:cw_bitcoin/bitcoin_address_record.dart';
|
||||||
import 'package:cw_bitcoin/bitcoin_transaction_credentials.dart';
|
import 'package:cw_bitcoin/bitcoin_transaction_credentials.dart';
|
||||||
import 'package:cw_bitcoin/bitcoin_transaction_no_inputs_exception.dart';
|
import 'package:cw_bitcoin/bitcoin_transaction_no_inputs_exception.dart';
|
||||||
import 'package:cw_bitcoin/bitcoin_transaction_priority.dart';
|
import 'package:cw_bitcoin/bitcoin_transaction_priority.dart';
|
||||||
|
import 'package:cw_bitcoin/bitcoin_transaction_wrong_balance_exception.dart';
|
||||||
import 'package:cw_bitcoin/bitcoin_unspent.dart';
|
import 'package:cw_bitcoin/bitcoin_unspent.dart';
|
||||||
import 'package:cw_bitcoin/electrum_balance.dart';
|
import 'package:cw_bitcoin/electrum_balance.dart';
|
||||||
import 'package:cw_bitcoin/electrum_wallet.dart';
|
import 'package:cw_bitcoin/electrum_wallet.dart';
|
||||||
|
@ -110,64 +111,149 @@ abstract class BitcoinCashWalletBase extends ElectrumWallet with Store {
|
||||||
Future<PendingBitcoinCashTransaction> createTransaction(Object credentials) async {
|
Future<PendingBitcoinCashTransaction> createTransaction(Object credentials) async {
|
||||||
const minAmount = 546;
|
const minAmount = 546;
|
||||||
final transactionCredentials = credentials as BitcoinTransactionCredentials;
|
final transactionCredentials = credentials as BitcoinTransactionCredentials;
|
||||||
|
final inputs = <BitcoinUnspent>[];
|
||||||
final outputs = transactionCredentials.outputs;
|
final outputs = transactionCredentials.outputs;
|
||||||
final hasMultiDestination = outputs.length > 1;
|
final hasMultiDestination = outputs.length > 1;
|
||||||
final builder = bitbox.Bitbox.transactionBuilder(testnet: false);
|
|
||||||
|
|
||||||
var allInputsAmount = 0;
|
var allInputsAmount = 0;
|
||||||
var inputs = <BitcoinUnspent>[];
|
|
||||||
|
|
||||||
if (unspentCoins.isEmpty) await updateUnspent();
|
if (unspentCoins.isEmpty) await updateUnspent();
|
||||||
|
|
||||||
inputs = unspentCoins.where((element) => element.isSending).toList();
|
for (final utx in unspentCoins) {
|
||||||
allInputsAmount = inputs.fold(0, (prev, element) => prev + element.value);
|
if (utx.isSending) {
|
||||||
|
allInputsAmount += utx.value;
|
||||||
|
inputs.add(utx);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (inputs.isEmpty) throw BitcoinTransactionNoInputsException();
|
if (inputs.isEmpty) throw BitcoinTransactionNoInputsException();
|
||||||
|
|
||||||
inputs.forEach((BitcoinUnspent utx) => builder.addInput(utx.hash, utx.vout));
|
|
||||||
|
|
||||||
final allAmountFee = transactionCredentials.feeRate != null
|
final allAmountFee = transactionCredentials.feeRate != null
|
||||||
? feeAmountWithFeeRate(transactionCredentials.feeRate!, inputs.length, outputs.length)
|
? feeAmountWithFeeRate(transactionCredentials.feeRate!, inputs.length, outputs.length)
|
||||||
: feeAmountForPriority(transactionCredentials.priority!, inputs.length, outputs.length);
|
: feeAmountForPriority(transactionCredentials.priority!, inputs.length, outputs.length);
|
||||||
|
|
||||||
final allAmount = allInputsAmount - allAmountFee;
|
final allAmount = allInputsAmount - allAmountFee;
|
||||||
|
|
||||||
|
var credentialsAmount = 0;
|
||||||
|
var amount = 0;
|
||||||
|
var fee = 0;
|
||||||
|
|
||||||
//allInputsAmount - transactionCredentials.outputs.fold(0, (prev, element) => prev + element.value);
|
if (hasMultiDestination) {
|
||||||
|
if (outputs.any((item) => item.sendAll || item.formattedCryptoAmount! <= 0)) {
|
||||||
|
throw BitcoinTransactionWrongBalanceException(currency);
|
||||||
|
}
|
||||||
|
|
||||||
|
credentialsAmount = outputs.fold(0, (acc, value) => acc + value.formattedCryptoAmount!);
|
||||||
|
|
||||||
// Calculate the amount to send and change
|
if (allAmount - credentialsAmount < minAmount) {
|
||||||
final sendAmount = transactionCredentials.outputs[0].formattedCryptoAmount!;
|
throw BitcoinTransactionWrongBalanceException(currency);
|
||||||
final outputAddress = transactionCredentials.outputs[0].isParsedAddress
|
}
|
||||||
? transactionCredentials.outputs[0].extractedAddress
|
|
||||||
: transactionCredentials.outputs[0].address;
|
|
||||||
final fee = bitbox.BitcoinCash.getByteCount(inputs.length, 2);
|
|
||||||
final changeAmount = allInputsAmount - sendAmount - fee;
|
|
||||||
|
|
||||||
// Add output for the recipient
|
amount = credentialsAmount;
|
||||||
builder.addOutput(outputAddress, sendAmount);
|
|
||||||
|
|
||||||
// Add change output if there is change
|
if (transactionCredentials.feeRate != null) {
|
||||||
if (changeAmount > 0) {
|
fee = calculateEstimatedFeeWithFeeRate(transactionCredentials.feeRate!, amount,
|
||||||
final changeAddress = await walletAddresses.getChangeAddress();
|
outputsCount: outputs.length + 1);
|
||||||
builder.addOutput(changeAddress, changeAmount);
|
} else {
|
||||||
|
fee = calculateEstimatedFee(transactionCredentials.priority, amount,
|
||||||
|
outputsCount: outputs.length + 1);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
final output = outputs.first;
|
||||||
|
credentialsAmount = !output.sendAll ? output.formattedCryptoAmount! : 0;
|
||||||
|
|
||||||
|
if (credentialsAmount > allAmount) {
|
||||||
|
throw BitcoinTransactionWrongBalanceException(currency);
|
||||||
|
}
|
||||||
|
|
||||||
|
amount = output.sendAll || allAmount - credentialsAmount < minAmount
|
||||||
|
? allAmount
|
||||||
|
: credentialsAmount;
|
||||||
|
|
||||||
|
if (output.sendAll || amount == allAmount) {
|
||||||
|
fee = allAmountFee;
|
||||||
|
} else if (transactionCredentials.feeRate != null) {
|
||||||
|
fee = calculateEstimatedFeeWithFeeRate(transactionCredentials.feeRate!, amount);
|
||||||
|
} else {
|
||||||
|
fee = calculateEstimatedFee(transactionCredentials.priority, amount);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (fee == 0) {
|
||||||
|
throw BitcoinTransactionWrongBalanceException(currency);
|
||||||
|
}
|
||||||
|
|
||||||
|
final totalAmount = amount + fee;
|
||||||
|
|
||||||
|
if (totalAmount > balance[currency]!.confirmed || totalAmount > allInputsAmount) {
|
||||||
|
throw BitcoinTransactionWrongBalanceException(currency);
|
||||||
|
}
|
||||||
|
final txb = bitbox.Bitbox.transactionBuilder(testnet: false);
|
||||||
|
|
||||||
|
final changeAddress = await walletAddresses.getChangeAddress();
|
||||||
|
var leftAmount = totalAmount;
|
||||||
|
var totalInputAmount = 0;
|
||||||
|
|
||||||
|
inputs.clear();
|
||||||
|
|
||||||
|
for (final utx in unspentCoins) {
|
||||||
|
if (utx.isSending) {
|
||||||
|
leftAmount = leftAmount - utx.value;
|
||||||
|
totalInputAmount += utx.value;
|
||||||
|
inputs.add(utx);
|
||||||
|
|
||||||
|
if (leftAmount <= 0) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (inputs.isEmpty) throw BitcoinTransactionNoInputsException();
|
||||||
|
|
||||||
|
if (amount <= 0 || totalInputAmount < totalAmount) {
|
||||||
|
throw BitcoinTransactionWrongBalanceException(currency);
|
||||||
|
}
|
||||||
|
|
||||||
|
inputs.forEach((input) {
|
||||||
|
txb.addInput(input.hash, input.vout);
|
||||||
|
});
|
||||||
|
|
||||||
|
outputs.forEach((item) {
|
||||||
|
final outputAmount = hasMultiDestination ? item.formattedCryptoAmount : amount;
|
||||||
|
final outputAddress = item.isParsedAddress ? item.extractedAddress! : item.address;
|
||||||
|
txb.addOutput(outputAddress, outputAmount!);
|
||||||
|
});
|
||||||
|
|
||||||
|
final estimatedSize = bitbox.BitcoinCash.getByteCount(inputs.length, outputs.length + 1);
|
||||||
|
|
||||||
|
var feeAmount = 0;
|
||||||
|
|
||||||
|
if (transactionCredentials.feeRate != null) {
|
||||||
|
feeAmount = transactionCredentials.feeRate! * estimatedSize;
|
||||||
|
} else {
|
||||||
|
feeAmount = feeRate(transactionCredentials.priority!) * estimatedSize;
|
||||||
|
}
|
||||||
|
|
||||||
|
final changeValue = totalInputAmount - amount - feeAmount;
|
||||||
|
|
||||||
|
if (changeValue > minAmount) {
|
||||||
|
txb.addOutput(changeAddress, changeValue);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Sign all inputs after adding all outputs
|
|
||||||
for (var i = 0; i < inputs.length; i++) {
|
for (var i = 0; i < inputs.length; i++) {
|
||||||
final input = inputs[i];
|
final input = inputs[i];
|
||||||
final keyPair = generateKeyPair(
|
final keyPair = generateKeyPair(
|
||||||
hd: input.bitcoinAddressRecord.isHidden ? walletAddresses.sideHd : walletAddresses.mainHd,
|
hd: input.bitcoinAddressRecord.isHidden ? walletAddresses.sideHd : walletAddresses.mainHd,
|
||||||
index: input.bitcoinAddressRecord.index,
|
index: input.bitcoinAddressRecord.index,
|
||||||
network: bitcoinCashNetworkType);
|
network: bitcoinCashNetworkType);
|
||||||
builder.sign(i, keyPair, input.value);
|
txb.sign(i, keyPair, input.value);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Build the transaction
|
// Build the transaction
|
||||||
final tx = builder.build();
|
final tx = txb.build();
|
||||||
|
|
||||||
return PendingBitcoinCashTransaction(tx, type,
|
return PendingBitcoinCashTransaction(tx, type,
|
||||||
electrumClient: electrumClient, amount: sendAmount, fee: fee);
|
electrumClient: electrumClient, amount: amount, fee: fee);
|
||||||
}
|
}
|
||||||
|
|
||||||
bitbox.ECPair generateKeyPair(
|
bitbox.ECPair generateKeyPair(
|
||||||
|
@ -177,12 +263,34 @@ abstract class BitcoinCashWalletBase extends ElectrumWallet with Store {
|
||||||
bitbox.ECPair.fromWIF(hd.derive(index).wif!);
|
bitbox.ECPair.fromWIF(hd.derive(index).wif!);
|
||||||
|
|
||||||
@override
|
@override
|
||||||
int feeAmountForPriority(BitcoinTransactionPriority priority, int inputsCount, int outputsCount) =>
|
int feeAmountForPriority(
|
||||||
|
BitcoinTransactionPriority priority, int inputsCount, int outputsCount) =>
|
||||||
feeRate(priority) * bitbox.BitcoinCash.getByteCount(inputsCount, outputsCount);
|
feeRate(priority) * bitbox.BitcoinCash.getByteCount(inputsCount, outputsCount);
|
||||||
|
|
||||||
int feeAmountWithFeeRate(int feeRate, int inputsCount, int outputsCount) =>
|
int feeAmountWithFeeRate(int feeRate, int inputsCount, int outputsCount) =>
|
||||||
feeRate * bitbox.BitcoinCash.getByteCount(inputsCount, outputsCount);
|
feeRate * bitbox.BitcoinCash.getByteCount(inputsCount, outputsCount);
|
||||||
|
|
||||||
|
int calculateEstimatedFeeWithFeeRate(int feeRate, int? amount, {int? outputsCount}) {
|
||||||
|
int inputsCount = 0;
|
||||||
|
int totalValue = 0;
|
||||||
|
|
||||||
|
for (final input in unspentCoins) {
|
||||||
|
if (input.isSending) {
|
||||||
|
inputsCount++;
|
||||||
|
totalValue += input.value;
|
||||||
|
}
|
||||||
|
if (amount != null && totalValue >= amount) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (amount != null && totalValue < amount) return 0;
|
||||||
|
|
||||||
|
final _outputsCount = outputsCount ?? (amount != null ? 2 : 1);
|
||||||
|
|
||||||
|
return feeAmountWithFeeRate(feeRate, inputsCount, _outputsCount);
|
||||||
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
int feeRate(TransactionPriority priority) {
|
int feeRate(TransactionPriority priority) {
|
||||||
if (priority is BitcoinCashTransactionPriority) {
|
if (priority is BitcoinCashTransactionPriority) {
|
||||||
|
@ -199,4 +307,3 @@ abstract class BitcoinCashWalletBase extends ElectrumWallet with Store {
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -48,10 +48,6 @@ class CWBitcoin extends Bitcoin {
|
||||||
List<TransactionPriority> getLitecoinTransactionPriorities()
|
List<TransactionPriority> getLitecoinTransactionPriorities()
|
||||||
=> LitecoinTransactionPriority.all;
|
=> LitecoinTransactionPriority.all;
|
||||||
|
|
||||||
@override
|
|
||||||
List<TransactionPriority> getBitcoinCashTransactionPriorities()
|
|
||||||
=> BitcoinTransactionPriority.all;
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
TransactionPriority deserializeBitcoinTransactionPriority(int raw)
|
TransactionPriority deserializeBitcoinTransactionPriority(int raw)
|
||||||
=> BitcoinTransactionPriority.deserialize(raw: raw);
|
=> BitcoinTransactionPriority.deserialize(raw: raw);
|
||||||
|
@ -60,10 +56,6 @@ class CWBitcoin extends Bitcoin {
|
||||||
TransactionPriority deserializeLitecoinTransactionPriority(int raw)
|
TransactionPriority deserializeLitecoinTransactionPriority(int raw)
|
||||||
=> LitecoinTransactionPriority.deserialize(raw: raw);
|
=> LitecoinTransactionPriority.deserialize(raw: raw);
|
||||||
|
|
||||||
@override
|
|
||||||
TransactionPriority deserializeBitcoinCashTransactionPriority(int raw)
|
|
||||||
=> BitcoinTransactionPriority.deserialize(raw: raw);
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
int getFeeRate(Object wallet, TransactionPriority priority) {
|
int getFeeRate(Object wallet, TransactionPriority priority) {
|
||||||
final bitcoinWallet = wallet as ElectrumWallet;
|
final bitcoinWallet = wallet as ElectrumWallet;
|
||||||
|
|
|
@ -28,11 +28,11 @@ class CWBitcoinCash extends BitcoinCash {
|
||||||
|
|
||||||
@override
|
@override
|
||||||
TransactionPriority deserializeBitcoinCashTransactionPriority(int raw) =>
|
TransactionPriority deserializeBitcoinCashTransactionPriority(int raw) =>
|
||||||
BitcoinTransactionPriority.deserialize(raw: raw);
|
BitcoinCashTransactionPriority.deserialize(raw: raw);
|
||||||
|
|
||||||
@override
|
@override
|
||||||
TransactionPriority getDefaultTransactionPriority() => BitcoinTransactionPriority.medium;
|
TransactionPriority getDefaultTransactionPriority() => BitcoinCashTransactionPriority.medium;
|
||||||
|
|
||||||
@override
|
@override
|
||||||
List<TransactionPriority> getTransactionPriorities() => BitcoinTransactionPriority.all;
|
List<TransactionPriority> getTransactionPriorities() => BitcoinCashTransactionPriority.all;
|
||||||
}
|
}
|
||||||
|
|
|
@ -731,7 +731,7 @@ abstract class SettingsStoreBase with Store {
|
||||||
priority[WalletType.ethereum]!;
|
priority[WalletType.ethereum]!;
|
||||||
}
|
}
|
||||||
if (sharedPreferences.getInt(PreferencesKey.bitcoinCashTransactionPriority) != null) {
|
if (sharedPreferences.getInt(PreferencesKey.bitcoinCashTransactionPriority) != null) {
|
||||||
priority[WalletType.bitcoinCash] = bitcoin?.deserializeBitcoinCashTransactionPriority(
|
priority[WalletType.bitcoinCash] = bitcoinCash?.deserializeBitcoinCashTransactionPriority(
|
||||||
sharedPreferences.getInt(PreferencesKey.bitcoinCashTransactionPriority)!) ??
|
sharedPreferences.getInt(PreferencesKey.bitcoinCashTransactionPriority)!) ??
|
||||||
priority[WalletType.bitcoinCash]!;
|
priority[WalletType.bitcoinCash]!;
|
||||||
}
|
}
|
||||||
|
|
|
@ -28,7 +28,6 @@ Future<void> main(List<String> args) async {
|
||||||
Future<void> generateBitcoin(bool hasImplementation) async {
|
Future<void> generateBitcoin(bool hasImplementation) async {
|
||||||
final outputFile = File(bitcoinOutputPath);
|
final outputFile = File(bitcoinOutputPath);
|
||||||
const bitcoinCommonHeaders = """
|
const bitcoinCommonHeaders = """
|
||||||
import 'package:cw_bitcoin_cash/cw_bitcoin_cash.dart';
|
|
||||||
import 'package:cw_core/wallet_credentials.dart';
|
import 'package:cw_core/wallet_credentials.dart';
|
||||||
import 'package:cw_core/wallet_info.dart';
|
import 'package:cw_core/wallet_info.dart';
|
||||||
import 'package:cw_core/transaction_priority.dart';
|
import 'package:cw_core/transaction_priority.dart';
|
||||||
|
@ -61,10 +60,8 @@ abstract class Bitcoin {
|
||||||
Map<String, String> getWalletKeys(Object wallet);
|
Map<String, String> getWalletKeys(Object wallet);
|
||||||
List<TransactionPriority> getTransactionPriorities();
|
List<TransactionPriority> getTransactionPriorities();
|
||||||
List<TransactionPriority> getLitecoinTransactionPriorities();
|
List<TransactionPriority> getLitecoinTransactionPriorities();
|
||||||
List<TransactionPriority> getBitcoinCashTransactionPriorities();
|
|
||||||
TransactionPriority deserializeBitcoinTransactionPriority(int raw);
|
TransactionPriority deserializeBitcoinTransactionPriority(int raw);
|
||||||
TransactionPriority deserializeLitecoinTransactionPriority(int raw);
|
TransactionPriority deserializeLitecoinTransactionPriority(int raw);
|
||||||
TransactionPriority deserializeBitcoinCashTransactionPriority(int raw);
|
|
||||||
int getFeeRate(Object wallet, TransactionPriority priority);
|
int getFeeRate(Object wallet, TransactionPriority priority);
|
||||||
Future<void> generateNewAddress(Object wallet);
|
Future<void> generateNewAddress(Object wallet);
|
||||||
Object createBitcoinTransactionCredentials(List<Output> outputs, {required TransactionPriority priority, int? feeRate});
|
Object createBitcoinTransactionCredentials(List<Output> outputs, {required TransactionPriority priority, int? feeRate});
|
||||||
|
|
Loading…
Reference in a new issue