167 lines
4.8 KiB
C
167 lines
4.8 KiB
C
|
// Copyright (c) 2012-2015, The CryptoNote developers, The Bytecoin developers
|
||
|
//
|
||
|
// This file is part of Bytecoin.
|
||
|
//
|
||
|
// Bytecoin is free software: you can redistribute it and/or modify
|
||
|
// it under the terms of the GNU Lesser General Public License as published by
|
||
|
// the Free Software Foundation, either version 3 of the License, or
|
||
|
// (at your option) any later version.
|
||
|
//
|
||
|
// Bytecoin is distributed in the hope that it will be useful,
|
||
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||
|
// GNU Lesser General Public License for more details.
|
||
|
//
|
||
|
// You should have received a copy of the GNU Lesser General Public License
|
||
|
// along with Bytecoin. If not, see <http://www.gnu.org/licenses/>.
|
||
|
|
||
|
#pragma once
|
||
|
|
||
|
#include "IWallet.h"
|
||
|
#include <atomic>
|
||
|
#include <condition_variable>
|
||
|
#include <deque>
|
||
|
#include <mutex>
|
||
|
#include <unordered_map>
|
||
|
|
||
|
namespace CryptoNote {
|
||
|
|
||
|
class WalletObserver: public IWalletObserver {
|
||
|
public:
|
||
|
|
||
|
WalletObserver() :
|
||
|
m_actualBalance(0),
|
||
|
m_actualBalancePrev(0),
|
||
|
m_pendingBalance(0),
|
||
|
m_pendingBalancePrev(0),
|
||
|
m_syncCount(0) {}
|
||
|
|
||
|
virtual void actualBalanceUpdated(uint64_t actualBalance) {
|
||
|
std::unique_lock<std::mutex> lk(m_mutex);
|
||
|
m_actualBalance = actualBalance;
|
||
|
lk.unlock();
|
||
|
m_cv.notify_all();
|
||
|
}
|
||
|
|
||
|
virtual void pendingBalanceUpdated(uint64_t pendingBalance) {
|
||
|
std::unique_lock<std::mutex> lk(m_mutex);
|
||
|
m_pendingBalance = pendingBalance;
|
||
|
lk.unlock();
|
||
|
m_cv.notify_all();
|
||
|
}
|
||
|
|
||
|
virtual void sendTransactionCompleted(CryptoNote::TransactionId transactionId, std::error_code result) {
|
||
|
std::unique_lock<std::mutex> lk(m_mutex);
|
||
|
m_sendResults[transactionId] = result;
|
||
|
m_cv.notify_all();
|
||
|
}
|
||
|
|
||
|
virtual void synchronizationCompleted(std::error_code result) {
|
||
|
std::unique_lock<std::mutex> lk(m_mutex);
|
||
|
++m_syncCount;
|
||
|
m_cv.notify_all();
|
||
|
}
|
||
|
|
||
|
virtual void synchronizationProgressUpdated(uint64_t current, uint64_t total) {
|
||
|
std::unique_lock<std::mutex> lk(m_mutex);
|
||
|
m_currentHeight = current;
|
||
|
m_cv.notify_all();
|
||
|
}
|
||
|
|
||
|
virtual void externalTransactionCreated(TransactionId transactionId) override {
|
||
|
std::unique_lock<std::mutex> lk(m_mutex);
|
||
|
m_externalTransactions.push_back(transactionId);
|
||
|
m_cv.notify_all();
|
||
|
}
|
||
|
|
||
|
uint64_t getCurrentHeight() {
|
||
|
std::unique_lock<std::mutex> lk(m_mutex);
|
||
|
return m_currentHeight;
|
||
|
}
|
||
|
|
||
|
uint64_t waitPendingBalanceChange() {
|
||
|
std::unique_lock<std::mutex> lk(m_mutex);
|
||
|
while (m_pendingBalance == m_pendingBalancePrev) {
|
||
|
m_cv.wait(lk);
|
||
|
}
|
||
|
m_pendingBalancePrev = m_pendingBalance;
|
||
|
return m_pendingBalance;
|
||
|
}
|
||
|
|
||
|
uint64_t waitTotalBalanceChange() {
|
||
|
std::unique_lock<std::mutex> lk(m_mutex);
|
||
|
while (m_pendingBalance == m_pendingBalancePrev && m_actualBalance == m_actualBalancePrev) {
|
||
|
m_cv.wait(lk);
|
||
|
}
|
||
|
|
||
|
m_actualBalancePrev = m_actualBalance;
|
||
|
m_pendingBalancePrev = m_pendingBalance;
|
||
|
|
||
|
return m_actualBalance + m_pendingBalance;
|
||
|
}
|
||
|
|
||
|
CryptoNote::TransactionId waitExternalTransaction() {
|
||
|
std::unique_lock<std::mutex> lk(m_mutex);
|
||
|
|
||
|
while (m_externalTransactions.empty()) {
|
||
|
m_cv.wait(lk);
|
||
|
}
|
||
|
|
||
|
CryptoNote::TransactionId txId = m_externalTransactions.front();
|
||
|
m_externalTransactions.pop_front();
|
||
|
return txId;
|
||
|
}
|
||
|
|
||
|
template<class Rep, class Period>
|
||
|
std::pair<bool, uint64_t> waitPendingBalanceChangeFor(const std::chrono::duration<Rep, Period>& timePeriod) {
|
||
|
std::unique_lock<std::mutex> lk(m_mutex);
|
||
|
bool result = m_cv.wait_for(lk, timePeriod, [&] { return m_pendingBalance != m_pendingBalancePrev; });
|
||
|
m_pendingBalancePrev = m_pendingBalance;
|
||
|
return std::make_pair(result, m_pendingBalance);
|
||
|
}
|
||
|
|
||
|
uint64_t waitActualBalanceChange() {
|
||
|
std::unique_lock<std::mutex> lk(m_mutex);
|
||
|
while (m_actualBalance == m_actualBalancePrev) {
|
||
|
m_cv.wait(lk);
|
||
|
}
|
||
|
m_actualBalancePrev = m_actualBalance;
|
||
|
return m_actualBalance;
|
||
|
}
|
||
|
|
||
|
std::error_code waitSendResult(CryptoNote::TransactionId txid) {
|
||
|
std::unique_lock<std::mutex> lk(m_mutex);
|
||
|
|
||
|
std::unordered_map<CryptoNote::TransactionId, std::error_code>::iterator it;
|
||
|
|
||
|
while ((it = m_sendResults.find(txid)) == m_sendResults.end()) {
|
||
|
m_cv.wait(lk);
|
||
|
}
|
||
|
|
||
|
return it->second;
|
||
|
}
|
||
|
|
||
|
uint64_t totalBalance() {
|
||
|
std::unique_lock<std::mutex> lk(m_mutex);
|
||
|
m_pendingBalancePrev = m_pendingBalance;
|
||
|
m_actualBalancePrev = m_actualBalance;
|
||
|
return m_pendingBalance + m_actualBalance;
|
||
|
}
|
||
|
|
||
|
private:
|
||
|
|
||
|
std::mutex m_mutex;
|
||
|
std::condition_variable m_cv;
|
||
|
uint64_t m_actualBalance;
|
||
|
uint64_t m_actualBalancePrev;
|
||
|
uint64_t m_pendingBalance;
|
||
|
uint64_t m_pendingBalancePrev;
|
||
|
size_t m_syncCount;
|
||
|
uint64_t m_currentHeight;
|
||
|
|
||
|
std::unordered_map<CryptoNote::TransactionId, std::error_code> m_sendResults;
|
||
|
std::deque<CryptoNote::TransactionId> m_externalTransactions;
|
||
|
};
|
||
|
|
||
|
}
|