localHistoryStorage #5
@ -261,7 +261,7 @@ std::list<Shared::Message> Core::Archive::getBefore(int count, const QString& id
|
|||||||
rc = mdb_cursor_open(txn, order, &cursor);
|
rc = mdb_cursor_open(txn, order, &cursor);
|
||||||
rc = mdb_cursor_get(cursor, &lmdbKey, &lmdbData, MDB_LAST);
|
rc = mdb_cursor_get(cursor, &lmdbKey, &lmdbData, MDB_LAST);
|
||||||
if (rc) {
|
if (rc) {
|
||||||
qDebug() << "Error geting before newest newest " << mdb_strerror(rc);
|
qDebug() << "Error geting before " << mdb_strerror(rc);
|
||||||
mdb_cursor_close(cursor);
|
mdb_cursor_close(cursor);
|
||||||
mdb_txn_abort(txn);
|
mdb_txn_abort(txn);
|
||||||
throw new Empty(jid.toStdString());
|
throw new Empty(jid.toStdString());
|
||||||
|
151
core/contact.cpp
151
core/contact.cpp
@ -31,7 +31,8 @@ Core::Contact::Contact(const QString& pJid, const QString& account, QObject* par
|
|||||||
receivedCount(0),
|
receivedCount(0),
|
||||||
hisoryCache(),
|
hisoryCache(),
|
||||||
appendCache(),
|
appendCache(),
|
||||||
requestCache()
|
requestCache(),
|
||||||
|
responseCache()
|
||||||
{
|
{
|
||||||
archive->open(account);
|
archive->open(account);
|
||||||
if (archive->isFromTheBeginning()) {
|
if (archive->isFromTheBeginning()) {
|
||||||
@ -115,19 +116,121 @@ void Core::Contact::requestHistory(int count, const QString& before)
|
|||||||
if (syncronizing) {
|
if (syncronizing) {
|
||||||
requestCache.emplace_back(count, before);
|
requestCache.emplace_back(count, before);
|
||||||
} else {
|
} else {
|
||||||
|
performRequest(count, before);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void Core::Contact::nextRequest()
|
||||||
|
{
|
||||||
|
if (requestCache.size() > 0) {
|
||||||
|
std::pair<int, QString> request = requestCache.front();
|
||||||
|
requestCache.pop_front();
|
||||||
|
performRequest(request.first, request.second);
|
||||||
|
} else {
|
||||||
|
syncronizing = false;
|
||||||
|
requestedCount = 0;
|
||||||
|
receivedCount = 0;
|
||||||
|
hisoryCache.clear();
|
||||||
|
responseCache.clear();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void Core::Contact::performRequest(int count, const QString& before)
|
||||||
|
{
|
||||||
|
syncronizing = true;
|
||||||
|
requestedCount = count;
|
||||||
|
receivedCount = 0;
|
||||||
|
hisoryCache.clear();
|
||||||
|
responseCache.clear();
|
||||||
|
|
||||||
switch (archiveState) {
|
switch (archiveState) {
|
||||||
case empty:
|
case empty:
|
||||||
|
emit needEarlierHistory(before, "", QDateTime(), QDateTime());
|
||||||
|
break;
|
||||||
|
case beginning: {
|
||||||
|
QString lBefore;
|
||||||
|
bool found = false;
|
||||||
if (appendCache.size() != 0) {
|
if (appendCache.size() != 0) {
|
||||||
//from last
|
if (before.size() == 0) {
|
||||||
|
lBefore = appendCache.back().getId();
|
||||||
|
}
|
||||||
|
for (std::list<Shared::Message>::const_reverse_iterator itr = appendCache.rbegin(), end = appendCache.rend(); itr != end; ++itr) {
|
||||||
|
const Shared::Message& msg = *itr;
|
||||||
|
if (found) {
|
||||||
|
responseCache.emplace_front(msg);
|
||||||
|
++receivedCount;
|
||||||
} else {
|
} else {
|
||||||
//search
|
if (msg.getId() == before) {
|
||||||
|
found = true;
|
||||||
|
responseCache.emplace_front(*itr);
|
||||||
|
++receivedCount;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (responseCache.size() == count) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (responseCache.size() == count) {
|
||||||
|
emit historyResponse(responseCache);
|
||||||
|
nextRequest();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (found) {
|
||||||
|
lBefore = responseCache.front().getId();
|
||||||
|
emit needEarlierHistory(lBefore, "", QDateTime(), QDateTime());
|
||||||
|
} else {
|
||||||
|
std::list<Shared::Message> arc;
|
||||||
|
if (responseCache.size() > 0) {
|
||||||
|
lBefore = responseCache.front().getId();
|
||||||
|
} else {
|
||||||
|
lBefore = before;
|
||||||
|
}
|
||||||
|
if (count != -1) {
|
||||||
|
try {
|
||||||
|
arc = archive->getBefore(requestedCount - receivedCount, lBefore);
|
||||||
|
responseCache.insert(responseCache.begin(), arc.begin(), arc.end());
|
||||||
|
emit historyResponse(responseCache);
|
||||||
|
nextRequest();
|
||||||
|
} catch (Archive::NotFound e) {
|
||||||
|
requestCache.emplace_back(count, before);
|
||||||
|
requestedCount = -1;
|
||||||
|
emit needEarlierHistory("", archive->newestId(), QDateTime(), QDateTime());
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
try {
|
||||||
|
arc = archive->getBefore(1, lBefore);
|
||||||
|
//just do nothing since response is not required
|
||||||
|
nextRequest(); //may be even it's a signal that the history is now complete?
|
||||||
|
} catch (Archive::NotFound e) {
|
||||||
|
emit needEarlierHistory("", archive->newestId(), QDateTime(), QDateTime());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case beginning:
|
|
||||||
//from last
|
|
||||||
break;
|
|
||||||
case end:
|
case end:
|
||||||
//try
|
std::list<Shared::Message> arc;
|
||||||
|
if (count != -1) {
|
||||||
|
try {
|
||||||
|
arc = archive->getBefore(requestedCount - receivedCount, before);
|
||||||
|
responseCache.insert(responseCache.begin(), arc.begin(), arc.end());
|
||||||
|
emit historyResponse(responseCache);
|
||||||
|
nextRequest();
|
||||||
|
} catch (Archive::NotFound e) {
|
||||||
|
requestCache.emplace_back(count, before);
|
||||||
|
requestedCount = -1;
|
||||||
|
emit needEarlierHistory("", archive->newestId(), QDateTime(), QDateTime());
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
try {
|
||||||
|
arc = archive->getBefore(1, before);
|
||||||
|
//just do nothing since response is not required
|
||||||
|
nextRequest(); //may be even it's a signal that the history is now complete?
|
||||||
|
} catch (Archive::NotFound e) {
|
||||||
|
emit needEarlierHistory("", archive->newestId(), QDateTime(), QDateTime());
|
||||||
|
}
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
case chunk:
|
case chunk:
|
||||||
//from last
|
//from last
|
||||||
@ -136,10 +239,34 @@ void Core::Contact::requestHistory(int count, const QString& before)
|
|||||||
//just give
|
//just give
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
syncronizing = true;
|
|
||||||
requestedCount = count;
|
|
||||||
receivedCount = 0;
|
|
||||||
hisoryCache.clear();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void Core::Contact::appendMessageToArchive(const Shared::Message& msg)
|
||||||
|
{
|
||||||
|
if (msg.getId().size() > 0 && msg.getBody().size() > 0) {
|
||||||
|
switch (archiveState) {
|
||||||
|
case empty:
|
||||||
|
if (archive->addElement(msg)) {
|
||||||
|
archiveState = end;
|
||||||
|
};
|
||||||
|
requestHistory(-1, msg.getId());
|
||||||
|
break;
|
||||||
|
case beginning:
|
||||||
|
appendCache.emplace_back(msg);
|
||||||
|
requestHistory(-1, msg.getId());
|
||||||
|
break;
|
||||||
|
case end:
|
||||||
|
archive->addElement(msg);
|
||||||
|
break;
|
||||||
|
case chunk:
|
||||||
|
appendCache.emplace_back(msg);
|
||||||
|
requestHistory(-1, msg.getId());
|
||||||
|
break;
|
||||||
|
case complete:
|
||||||
|
archive->addElement(msg);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -60,7 +60,7 @@ signals:
|
|||||||
void nameChanged(const QString& name);
|
void nameChanged(const QString& name);
|
||||||
void subscriptionStateChanged(Shared::SubscriptionState state);
|
void subscriptionStateChanged(Shared::SubscriptionState state);
|
||||||
void historyResponse(const std::list<Shared::Message>& messages);
|
void historyResponse(const std::list<Shared::Message>& messages);
|
||||||
void needEarlierHistory(const QString& before, const QString& after, int count, const QDateTime& from, const QDateTime& to);
|
void needEarlierHistory(const QString& before, const QString& after, const QDateTime& from, const QDateTime& to);
|
||||||
|
|
||||||
public:
|
public:
|
||||||
const QString jid;
|
const QString jid;
|
||||||
@ -77,7 +77,12 @@ private:
|
|||||||
int receivedCount;
|
int receivedCount;
|
||||||
std::list<Shared::Message> hisoryCache;
|
std::list<Shared::Message> hisoryCache;
|
||||||
std::list<Shared::Message> appendCache;
|
std::list<Shared::Message> appendCache;
|
||||||
|
std::list<Shared::Message> responseCache;
|
||||||
std::list<std::pair<int, QString>> requestCache;
|
std::list<std::pair<int, QString>> requestCache;
|
||||||
|
|
||||||
|
private:
|
||||||
|
void nextRequest();
|
||||||
|
void performRequest(int count, const QString& before);
|
||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user