void sendMultipleTransactions(CryptoNote::WalletGreen &wallet, std::vector<CryptoNote::TransactionParameters> transfers) { size_t numTxs = transfers.size(); size_t currentTx = 1; std::cout << "Your transaction has been split up into " << numTxs << " separate transactions of " << formatAmount(transfers[0].destinations[0].amount) << ". It may take some time to send all the transactions, " << "please be patient." << std::endl << std::endl; for (auto tx : transfers) { while (true) { std::cout << "Attempting to send transaction " << InformationMsg(std::to_string(currentTx)) << " of " << InformationMsg(std::to_string(numTxs)) << std::endl; wallet.updateInternalCache(); uint64_t neededBalance = tx.destinations[0].amount + tx.fee; if (neededBalance < wallet.getActualBalance()) { size_t id = wallet.transfer(tx); CryptoNote::WalletTransaction sentTx = wallet.getTransaction(id); std::cout << SuccessMsg("Transaction has been sent!") << std::endl << SuccessMsg("Hash: " + Common::podToHex(sentTx.hash)) << std::endl << std::endl; break; } std::cout << "Not enough balance available to send transaction, " << "this is because some of your balance is used when " << "sending another transaction to help hide the size " << "of your transaction, and is locked for a short " << "time. It will return shortly." << std::endl << "Needed balance: " << formatAmount(neededBalance) << std::endl << "Available balance: " << formatAmount(wallet.getActualBalance()) << std::endl << "Locked balance: " << formatAmount(wallet.getPendingBalance()) << std::endl << "Will try again in 5 seconds..." << std::endl << std::endl; std::this_thread::sleep_for(std::chrono::seconds(5)); } currentTx++; } std::cout << SuccessMsg("All transactions sent!") << std::endl; }
bool optimize(CryptoNote::WalletGreen &wallet, uint64_t threshold) { std::vector<Crypto::Hash> fusionTransactionHashes; while (true) { /* Create as many fusion transactions until we can't send anymore, either because balance is locked too much or we can no longer optimize anymore transactions */ const size_t tmpFusionTxID = makeFusionTransaction(wallet, threshold); if (tmpFusionTxID == CryptoNote::WALLET_INVALID_TRANSACTION_ID) { break; } else { const CryptoNote::WalletTransaction w = wallet.getTransaction(tmpFusionTxID); fusionTransactionHashes.push_back(w.hash); if (fusionTransactionHashes.size() == 1) { std::cout << SuccessMsg("Created 1 fusion transaction!") << std::endl; } else { std::cout << SuccessMsg("Created " + std::to_string(fusionTransactionHashes.size()) + " fusion transactions!") << std::endl; } } } if (fusionTransactionHashes.empty()) { return false; } /* Hurr durr grammar */ if (fusionTransactionHashes.size() == 1) { std::cout << SuccessMsg("1 fusion transaction has been sent, waiting " "for balance to return and unlock") << std::endl << std::endl; } else { std::cout << SuccessMsg(std::to_string(fusionTransactionHashes.size()) + " fusion transactions have been sent, waiting " "for balance to return and unlock") << std::endl << std::endl; } wallet.updateInternalCache(); /* Short sleep to ensure it's in the transaction pool when we poll it */ std::this_thread::sleep_for(std::chrono::seconds(1)); while (true) { const std::vector<CryptoNote::WalletTransactionWithTransfers> unconfirmedTransactions = wallet.getUnconfirmedTransactions(); std::vector<Crypto::Hash> unconfirmedTxHashes; for (const auto &t : unconfirmedTransactions) { unconfirmedTxHashes.push_back(t.transaction.hash); } bool fusionCompleted = true; /* Is our fusion transaction still unconfirmed? We can't gain the benefits of fusioning if the balance hasn't unlocked, so we can send this new optimized balance */ for (const auto &tx : fusionTransactionHashes) { /* If the fusion transaction hash is present in the unconfirmed transactions pool, we need to wait for it to complete. */ if (std::find(unconfirmedTxHashes.begin(), unconfirmedTxHashes.end(), tx) != unconfirmedTxHashes.end()) { fusionCompleted = false; } else { /* We can't find this transaction in the unconfirmed transaction pool anymore, so it has been confirmed. Remove it so we both have to check less transactions each time, and we can easily update the transactions left to confirm output message */ fusionTransactionHashes.erase(std::remove (fusionTransactionHashes.begin(), fusionTransactionHashes.end(), tx), fusionTransactionHashes.end()); } } if (!fusionCompleted) { std::cout << WarningMsg("Balance is still locked, " + std::to_string(fusionTransactionHashes.size())); /* More grammar... */ if (fusionTransactionHashes.size() == 1) { std::cout << WarningMsg(" fusion transaction still to be " "confirmed."); } else { std::cout << WarningMsg(" fusion transactions still to be " "confirmed."); } std::cout << std::endl << SuccessMsg("Will try again in 1 minute...") << std::endl; std::this_thread::sleep_for(std::chrono::minutes(1)); wallet.updateInternalCache(); } else { std::cout << SuccessMsg("All fusion transactions confirmed!") << std::endl; break; } } return true; }