mirror of
https://github.com/monero-project/monero-gui.git
synced 2025-01-25 12:05:54 +00:00
TransactionHistory: guard tx info list against concurrent access
This commit is contained in:
parent
d5f4d5d93f
commit
9d5eb002ae
2 changed files with 45 additions and 31 deletions
|
@ -32,10 +32,13 @@
|
||||||
|
|
||||||
#include <QFile>
|
#include <QFile>
|
||||||
#include <QDebug>
|
#include <QDebug>
|
||||||
|
#include <QReadLocker>
|
||||||
|
#include <QWriteLocker>
|
||||||
|
|
||||||
|
|
||||||
TransactionInfo *TransactionHistory::transaction(int index)
|
TransactionInfo *TransactionHistory::transaction(int index)
|
||||||
{
|
{
|
||||||
|
QReadLocker locker(&m_tinfoLock);
|
||||||
|
|
||||||
if (index < 0 || index >= m_tinfo.size()) {
|
if (index < 0 || index >= m_tinfo.size()) {
|
||||||
qCritical("%s: no transaction info for index %d", __FUNCTION__, index);
|
qCritical("%s: no transaction info for index %d", __FUNCTION__, index);
|
||||||
|
@ -53,41 +56,48 @@ TransactionInfo *TransactionHistory::transaction(int index)
|
||||||
|
|
||||||
QList<TransactionInfo *> TransactionHistory::getAll(quint32 accountIndex) const
|
QList<TransactionInfo *> TransactionHistory::getAll(quint32 accountIndex) const
|
||||||
{
|
{
|
||||||
// XXX this invalidates previously saved history that might be used by model
|
|
||||||
emit refreshStarted();
|
|
||||||
qDeleteAll(m_tinfo);
|
|
||||||
m_tinfo.clear();
|
|
||||||
|
|
||||||
QDateTime firstDateTime = QDateTime(QDate(2014, 4, 18)); // the genesis block
|
QDateTime firstDateTime = QDateTime(QDate(2014, 4, 18)); // the genesis block
|
||||||
QDateTime lastDateTime = QDateTime::currentDateTime().addDays(1); // tomorrow (guard against jitter and timezones)
|
QDateTime lastDateTime = QDateTime::currentDateTime().addDays(1); // tomorrow (guard against jitter and timezones)
|
||||||
quint64 lastTxHeight = 0;
|
|
||||||
m_locked = false;
|
|
||||||
m_minutesToUnlock = 0;
|
|
||||||
TransactionHistory * parent = const_cast<TransactionHistory*>(this);
|
|
||||||
for (const auto i : m_pimpl->getAll()) {
|
|
||||||
TransactionInfo * ti = new TransactionInfo(i, parent);
|
|
||||||
if (ti->subaddrAccount() != accountIndex) {
|
|
||||||
delete ti;
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
m_tinfo.append(ti);
|
|
||||||
// looking for transactions timestamp scope
|
|
||||||
if (ti->timestamp() >= lastDateTime) {
|
|
||||||
lastDateTime = ti->timestamp();
|
|
||||||
}
|
|
||||||
if (ti->timestamp() <= firstDateTime) {
|
|
||||||
firstDateTime = ti->timestamp();
|
|
||||||
}
|
|
||||||
quint64 requiredConfirmations = (ti->blockHeight() < ti->unlockTime()) ? ti->unlockTime() - ti->blockHeight() : 10;
|
|
||||||
// store last tx height
|
|
||||||
if (ti->confirmations() < requiredConfirmations && ti->blockHeight() >= lastTxHeight) {
|
|
||||||
lastTxHeight = ti->blockHeight();
|
|
||||||
// TODO: Fetch block time and confirmations needed from wallet2?
|
|
||||||
m_minutesToUnlock = (requiredConfirmations - ti->confirmations()) * 2;
|
|
||||||
m_locked = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
emit refreshStarted();
|
||||||
|
|
||||||
|
{
|
||||||
|
QWriteLocker locker(&m_tinfoLock);
|
||||||
|
|
||||||
|
// XXX this invalidates previously saved history that might be used by model
|
||||||
|
qDeleteAll(m_tinfo);
|
||||||
|
m_tinfo.clear();
|
||||||
|
|
||||||
|
quint64 lastTxHeight = 0;
|
||||||
|
m_locked = false;
|
||||||
|
m_minutesToUnlock = 0;
|
||||||
|
TransactionHistory * parent = const_cast<TransactionHistory*>(this);
|
||||||
|
for (const auto i : m_pimpl->getAll()) {
|
||||||
|
TransactionInfo * ti = new TransactionInfo(i, parent);
|
||||||
|
if (ti->subaddrAccount() != accountIndex) {
|
||||||
|
delete ti;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
m_tinfo.append(ti);
|
||||||
|
// looking for transactions timestamp scope
|
||||||
|
if (ti->timestamp() >= lastDateTime) {
|
||||||
|
lastDateTime = ti->timestamp();
|
||||||
|
}
|
||||||
|
if (ti->timestamp() <= firstDateTime) {
|
||||||
|
firstDateTime = ti->timestamp();
|
||||||
|
}
|
||||||
|
quint64 requiredConfirmations = (ti->blockHeight() < ti->unlockTime()) ? ti->unlockTime() - ti->blockHeight() : 10;
|
||||||
|
// store last tx height
|
||||||
|
if (ti->confirmations() < requiredConfirmations && ti->blockHeight() >= lastTxHeight) {
|
||||||
|
lastTxHeight = ti->blockHeight();
|
||||||
|
// TODO: Fetch block time and confirmations needed from wallet2?
|
||||||
|
m_minutesToUnlock = (requiredConfirmations - ti->confirmations()) * 2;
|
||||||
|
m_locked = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
emit refreshFinished();
|
emit refreshFinished();
|
||||||
|
|
||||||
if (m_firstDateTime != firstDateTime) {
|
if (m_firstDateTime != firstDateTime) {
|
||||||
|
@ -112,6 +122,8 @@ void TransactionHistory::refresh(quint32 accountIndex)
|
||||||
|
|
||||||
quint64 TransactionHistory::count() const
|
quint64 TransactionHistory::count() const
|
||||||
{
|
{
|
||||||
|
QReadLocker locker(&m_tinfoLock);
|
||||||
|
|
||||||
return m_tinfo.count();
|
return m_tinfo.count();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -31,6 +31,7 @@
|
||||||
|
|
||||||
#include <QObject>
|
#include <QObject>
|
||||||
#include <QList>
|
#include <QList>
|
||||||
|
#include <QReadWriteLock>
|
||||||
#include <QDateTime>
|
#include <QDateTime>
|
||||||
|
|
||||||
namespace Monero {
|
namespace Monero {
|
||||||
|
@ -76,6 +77,7 @@ private:
|
||||||
friend class Wallet;
|
friend class Wallet;
|
||||||
Monero::TransactionHistory * m_pimpl;
|
Monero::TransactionHistory * m_pimpl;
|
||||||
mutable QList<TransactionInfo*> m_tinfo;
|
mutable QList<TransactionInfo*> m_tinfo;
|
||||||
|
mutable QReadWriteLock m_tinfoLock;
|
||||||
mutable QDateTime m_firstDateTime;
|
mutable QDateTime m_firstDateTime;
|
||||||
mutable QDateTime m_lastDateTime;
|
mutable QDateTime m_lastDateTime;
|
||||||
mutable int m_minutesToUnlock;
|
mutable int m_minutesToUnlock;
|
||||||
|
|
Loading…
Reference in a new issue