diff --git a/src/SimpleWallet/SubWallet.cpp b/src/SimpleWallet/SubWallet.cpp
index 9f280a9..e99facc 100644
--- a/src/SimpleWallet/SubWallet.cpp
+++ b/src/SimpleWallet/SubWallet.cpp
@@ -1,5 +1,5 @@
/*
-Copyright (C) 2021, The Talleo developers
+Copyright (C) 2021-2024, The Talleo developers
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
@@ -17,6 +17,16 @@ along with this program. If not, see .
#include "SubWallet.h"
+uint64_t getTotalActualBalance(CryptoNote::WalletGreen &wallet) {
+ uint64_t balance = 0;
+ size_t numWallets = wallet.getAddressCount();
+ for (size_t i = 0; i < numWallets; i++) {
+ std::string address = wallet.getAddress(i);
+ balance += wallet.getActualBalance(address);
+ }
+ return balance;
+}
+
uint64_t getTotalActualBalance(CryptoNote::WalletGreen &wallet, const std::vector &addresses) {
uint64_t balance = 0;
for (auto address : addresses) {
@@ -25,6 +35,16 @@ uint64_t getTotalActualBalance(CryptoNote::WalletGreen &wallet, const std::vecto
return balance;
}
+uint64_t getTotalPendingBalance(CryptoNote::WalletGreen &wallet) {
+ uint64_t balance = 0;
+ size_t numWallets = wallet.getAddressCount();
+ for (size_t i = 0; i < numWallets; i++) {
+ std::string address = wallet.getAddress(i);
+ balance += wallet.getPendingBalance(address);
+ }
+ return balance;
+}
+
uint64_t getTotalPendingBalance(CryptoNote::WalletGreen &wallet, const std::vector &addresses) {
uint64_t balance = 0;
for (auto address: addresses) {
diff --git a/src/SimpleWallet/SubWallet.h b/src/SimpleWallet/SubWallet.h
index 3ae9549..9bfcd33 100644
--- a/src/SimpleWallet/SubWallet.h
+++ b/src/SimpleWallet/SubWallet.h
@@ -1,5 +1,5 @@
/*
-Copyright (C) 2021, The Talleo developers
+Copyright (C) 2021-2024, The Talleo developers
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
@@ -19,5 +19,7 @@ along with this program. If not, see .
#include "Wallet/WalletGreen.h"
+uint64_t getTotalActualBalance(CryptoNote::WalletGreen &wallet);
uint64_t getTotalActualBalance(CryptoNote::WalletGreen &wallet, const std::vector &addresses);
+uint64_t getTotalPendingBalance(CryptoNote::WalletGreen &wallet);
uint64_t getTotalPendingBalance(CryptoNote::WalletGreen &wallet, const std::vector &addresses);
diff --git a/src/SimpleWallet/Transfer.cpp b/src/SimpleWallet/Transfer.cpp
index 68a94db..67d83dd 100644
--- a/src/SimpleWallet/Transfer.cpp
+++ b/src/SimpleWallet/Transfer.cpp
@@ -542,6 +542,9 @@ void fusionTX(CryptoNote::WalletGreen &wallet, CryptoNote::TransactionParameters
void transfer(System::Dispatcher& dispatcher, std::shared_ptr walletInfo, std::vector args) {
uint16_t mixin;
+ std::string sourceAddress = walletInfo->wallet.getAddress(subWallet);
+ std::vector sourceAddresses;
+ sourceAddresses.push_back(sourceAddress);
std::string address;
uint64_t amount;
uint64_t fee = CryptoNote::parameters::MINIMUM_FEE;
@@ -607,7 +610,7 @@ void transfer(System::Dispatcher& dispatcher, std::shared_ptr wallet
}
}
- doTransfer(dispatcher, mixin, address, amount, fee, extra, walletInfo);
+ doTransfer(dispatcher, mixin, sourceAddresses, address, amount, fee, extra, walletInfo);
}
void transfer(System::Dispatcher& dispatcher, std::shared_ptr walletInfo) {
@@ -616,6 +619,8 @@ void transfer(System::Dispatcher& dispatcher, std::shared_ptr wallet
std::string sourceAddress = walletInfo->wallet.getAddress(subWallet);
uint64_t balance = walletInfo->wallet.getActualBalance(sourceAddress);
+ std::vector sourceAddresses;
+ sourceAddresses.push_back(sourceAddress);
auto maybeAddress = getDestinationAddress(dispatcher);
@@ -635,6 +640,25 @@ void transfer(System::Dispatcher& dispatcher, std::shared_ptr wallet
uint64_t amount = maybeAmount.x;
+ if (balance < amount && getTotalActualBalance(walletInfo->wallet) > amount) {
+ std::cout << WarningMsg("Current subwallet doesn't have enough funds to cover this transaction,") << std::endl;
+ if (confirm("do you want to use other subwallets?")) {
+ size_t numWallets = walletInfo->wallet.getAddressCount();
+ for (size_t i = 0; i < numWallets; i++) {
+ std::string subaddress = walletInfo->wallet.getAddress(i);
+ uint64_t subbalance = walletInfo->wallet.getActualBalance(subaddress);
+ if (subaddress != sourceAddress && subbalance > 0) {
+ balance += subbalance;
+ sourceAddresses.push_back(subaddress);
+ // We need at least amount + 0.01 TLO
+ if (balance > amount) {
+ break;
+ }
+ }
+ }
+ }
+ }
+
if (balance < amount) {
size_t totalLen = formatAmount(amount).length(); // Funds needed is always the widest string
@@ -653,6 +677,27 @@ void transfer(System::Dispatcher& dispatcher, std::shared_ptr wallet
uint64_t fee = maybeFee.x;
+ if (balance < amount + fee && sourceAddresses.size() > 1 && getTotalActualBalance(walletInfo->wallet) >= amount + fee) {
+ size_t numWallets = walletInfo->wallet.getAddressCount();
+ uint64_t needed = amount + fee;
+ // We need to consider remote node fee already here as it could be more than remaining balance of already selected subwallets
+ if (!remote_fee_address.empty()) {
+ // Remote node fee is between 0.01 and 1.00 TLO depending on transfer amount
+ needed += std::min(UINT64_C(1), std::max(static_cast(amount * 0.000025), UINT64_C(100)));
+ }
+ for (size_t i = 0; i < numWallets; i++) {
+ std::string subaddress = walletInfo->wallet.getAddress(i);
+ uint64_t subbalance = walletInfo->wallet.getActualBalance(subaddress);
+ if (std::find(sourceAddresses.begin(), sourceAddresses.end(), subaddress) == sourceAddresses.end() && subbalance > 0) {
+ balance += subbalance;
+ sourceAddresses.push_back(subaddress);
+ if (balance >= needed) {
+ break;
+ }
+ }
+ }
+ }
+
if (balance < amount + fee) {
size_t totalLen = formatAmount(amount + fee).length(); // Funds needed is always the widest string
@@ -680,12 +725,12 @@ void transfer(System::Dispatcher& dispatcher, std::shared_ptr wallet
std::string extra = maybeExtra.x;
- doTransfer(dispatcher, mixin, address, amount, fee, extra, walletInfo);
+ doTransfer(dispatcher, mixin, sourceAddresses, address, amount, fee, extra, walletInfo);
}
-void doTransfer(System::Dispatcher& dispatcher, uint16_t mixin, std::string address, uint64_t amount, uint64_t fee, std::string extra, std::shared_ptr walletInfo) {
- std::string sourceAddress = walletInfo->wallet.getAddress(subWallet);
- uint64_t balance = walletInfo->wallet.getActualBalance(sourceAddress);
+void doTransfer(System::Dispatcher& dispatcher, uint16_t mixin, std::vector sourceAddresses, std::string address, uint64_t amount, uint64_t fee, std::string extra, std::shared_ptr walletInfo) {
+ std::string changeAddress = walletInfo->wallet.getAddress(subWallet);
+ uint64_t balance = getTotalActualBalance(walletInfo->wallet, sourceAddresses);
uint64_t remote_node_fee = 0;
if (!remote_fee_address.empty()) {
// Remote node fee is between 0.01 and 1.00 TLO depending on transfer amount
@@ -726,12 +771,12 @@ void doTransfer(System::Dispatcher& dispatcher, uint16_t mixin, std::string addr
}
CryptoNote::TransactionParameters p;
- p.sourceAddresses.push_back(sourceAddress);
+ p.sourceAddresses = sourceAddresses;
p.destinations = transfers;
p.fee = fee;
p.mixIn = mixin;
p.extra = extra;
- p.changeDestination = sourceAddress;
+ p.changeDestination = changeAddress;
if (!confirmTransaction(p, walletInfo)) {
std::cout << WarningMsg("Cancelling transaction.") << std::endl;
@@ -1088,4 +1133,4 @@ std::string resolveAddress(System::Dispatcher& dispatcher, const std::string& em
return address;
-}
\ No newline at end of file
+}
diff --git a/src/SimpleWallet/Transfer.h b/src/SimpleWallet/Transfer.h
index f14a59e..e684d38 100644
--- a/src/SimpleWallet/Transfer.h
+++ b/src/SimpleWallet/Transfer.h
@@ -58,8 +58,8 @@ void transfer(System::Dispatcher& dispatcher, std::shared_ptr wallet
void transfer(System::Dispatcher& dispatcher, std::shared_ptr walletInfo,
std::vector args);
-void doTransfer(System::Dispatcher& dispatcher, uint16_t mixin, std::string address, uint64_t amount,
- uint64_t fee, std::string extra,
+void doTransfer(System::Dispatcher& dispatcher, uint16_t mixin, std::vector sourceAddresses,
+ std::string address, uint64_t amount, uint64_t fee, std::string extra,
std::shared_ptr walletInfo);
void fusionTX(CryptoNote::WalletGreen &wallet,
@@ -107,4 +107,4 @@ Maybe getMixin();
size_t makeFusionTransaction(CryptoNote::WalletGreen &wallet, const std::vector &sourceAddresses,
const std::string &destinationAddress, uint64_t threshold);
-std::string resolveAddress(System::Dispatcher& dispatcher, const std::string& email);
\ No newline at end of file
+std::string resolveAddress(System::Dispatcher& dispatcher, const std::string& email);