mirror of https://github.com/procxx/kepka.git
Prepare for syncing read / write requests.
This commit is contained in:
parent
db322cc19a
commit
147e8cc467
|
@ -241,11 +241,21 @@ void Histories::sendDialogRequests() {
|
||||||
if (_dialogRequestsPending.empty()) {
|
if (_dialogRequestsPending.empty()) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
auto histories = std::vector<not_null<History*>>();
|
const auto histories = ranges::view::all(
|
||||||
ranges::transform(
|
_dialogRequestsPending
|
||||||
_dialogRequestsPending,
|
) | ranges::view::transform([](const auto &pair) {
|
||||||
ranges::back_inserter(histories),
|
return pair.first;
|
||||||
[](const auto &pair) { return pair.first; });
|
}) | ranges::view::filter([&](not_null<History*> history) {
|
||||||
|
const auto state = lookup(history);
|
||||||
|
if (!state) {
|
||||||
|
return true;
|
||||||
|
} else if (!postponeEntryRequest(*state)) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
state->postponedRequestEntry = true;
|
||||||
|
return false;
|
||||||
|
}) | ranges::to_vector;
|
||||||
|
|
||||||
auto peers = QVector<MTPInputDialogPeer>();
|
auto peers = QVector<MTPInputDialogPeer>();
|
||||||
const auto dialogPeer = [](not_null<History*> history) {
|
const auto dialogPeer = [](not_null<History*> history) {
|
||||||
return MTP_inputDialogPeer(history->peer->input);
|
return MTP_inputDialogPeer(history->peer->input);
|
||||||
|
@ -260,8 +270,11 @@ void Histories::sendDialogRequests() {
|
||||||
|
|
||||||
const auto finalize = [=] {
|
const auto finalize = [=] {
|
||||||
for (const auto history : histories) {
|
for (const auto history : histories) {
|
||||||
dialogEntryApplied(history);
|
const auto state = lookup(history);
|
||||||
history->updateChatListExistence();
|
if (!state || !state->postponedRequestEntry) {
|
||||||
|
dialogEntryApplied(history);
|
||||||
|
history->updateChatListExistence();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
session().api().request(MTPmessages_GetPeerDialogs(
|
session().api().request(MTPmessages_GetPeerDialogs(
|
||||||
|
@ -401,6 +414,7 @@ void Histories::sendReadRequest(not_null<History*> history, State &state) {
|
||||||
void Histories::checkEmptyState(not_null<History*> history) {
|
void Histories::checkEmptyState(not_null<History*> history) {
|
||||||
const auto empty = [](const State &state) {
|
const auto empty = [](const State &state) {
|
||||||
return state.postponed.empty()
|
return state.postponed.empty()
|
||||||
|
&& !state.postponedRequestEntry
|
||||||
&& state.sent.empty()
|
&& state.sent.empty()
|
||||||
&& (state.readTill == 0);
|
&& (state.readTill == 0);
|
||||||
};
|
};
|
||||||
|
@ -410,6 +424,21 @@ void Histories::checkEmptyState(not_null<History*> history) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool Histories::postponeHistoryRequest(const State &state) const {
|
||||||
|
const auto proj = [](const auto &pair) {
|
||||||
|
return pair.second.type;
|
||||||
|
};
|
||||||
|
const auto i = ranges::find(state.sent, RequestType::Delete, proj);
|
||||||
|
return (i != end(state.sent));
|
||||||
|
}
|
||||||
|
|
||||||
|
bool Histories::postponeEntryRequest(const State &state) const {
|
||||||
|
const auto i = ranges::find_if(state.sent, [](const auto &pair) {
|
||||||
|
return pair.second.type != RequestType::History;
|
||||||
|
});
|
||||||
|
return (i != end(state.sent));
|
||||||
|
}
|
||||||
|
|
||||||
int Histories::sendRequest(
|
int Histories::sendRequest(
|
||||||
not_null<History*> history,
|
not_null<History*> history,
|
||||||
RequestType type,
|
RequestType type,
|
||||||
|
@ -418,119 +447,85 @@ int Histories::sendRequest(
|
||||||
|
|
||||||
auto &state = _states[history];
|
auto &state = _states[history];
|
||||||
const auto id = ++state.autoincrement;
|
const auto id = ++state.autoincrement;
|
||||||
const auto action = chooseAction(state, type);
|
if (type == RequestType::History && postponeHistoryRequest(state)) {
|
||||||
if (action == Action::Send) {
|
|
||||||
state.sent.emplace(id, SentRequest{
|
|
||||||
generator([=] { checkPostponed(history, id); }),
|
|
||||||
type
|
|
||||||
});
|
|
||||||
if (base::take(state.thenRequestEntry)) {
|
|
||||||
requestDialogEntry(history);
|
|
||||||
}
|
|
||||||
} else if (action == Action::Postpone) {
|
|
||||||
state.postponed.emplace(
|
state.postponed.emplace(
|
||||||
id,
|
id,
|
||||||
PostponedRequest{ std::move(generator), type });
|
PostponedHistoryRequest{ std::move(generator) });
|
||||||
|
return id;
|
||||||
|
}
|
||||||
|
const auto requestId = generator([=] { checkPostponed(history, id); });
|
||||||
|
state.sent.emplace(id, SentRequest{
|
||||||
|
std::move(generator),
|
||||||
|
requestId,
|
||||||
|
type
|
||||||
|
});
|
||||||
|
if (!state.postponedRequestEntry
|
||||||
|
&& postponeEntryRequest(state)
|
||||||
|
&& _dialogRequests.contains(history)) {
|
||||||
|
state.postponedRequestEntry = true;
|
||||||
|
}
|
||||||
|
if (postponeHistoryRequest(state)) {
|
||||||
|
const auto resendHistoryRequest = [&](auto &pair) {
|
||||||
|
auto &[id, sent] = pair;
|
||||||
|
if (sent.type != RequestType::History) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
state.postponed.emplace(
|
||||||
|
id,
|
||||||
|
PostponedHistoryRequest{ std::move(sent.generator) });
|
||||||
|
session().api().request(sent.id).cancel();
|
||||||
|
return true;
|
||||||
|
};
|
||||||
|
state.sent.erase(
|
||||||
|
ranges::remove_if(state.sent, resendHistoryRequest),
|
||||||
|
end(state.sent));
|
||||||
}
|
}
|
||||||
return id;
|
return id;
|
||||||
}
|
}
|
||||||
|
|
||||||
void Histories::checkPostponed(not_null<History*> history, int requestId) {
|
void Histories::checkPostponed(not_null<History*> history, int id) {
|
||||||
const auto state = lookup(history);
|
const auto state = lookup(history);
|
||||||
Assert(state != nullptr);
|
Assert(state != nullptr);
|
||||||
|
|
||||||
state->sent.remove(requestId);
|
finishSentRequest(history, state, id);
|
||||||
if (!state->postponed.empty()) {
|
|
||||||
auto &entry = state->postponed.front();
|
|
||||||
const auto action = chooseAction(*state, entry.second.type, true);
|
|
||||||
if (action == Action::Send) {
|
|
||||||
const auto id = entry.first;
|
|
||||||
const auto postponed = std::move(entry.second);
|
|
||||||
state->postponed.remove(id);
|
|
||||||
state->sent.emplace(id, SentRequest{
|
|
||||||
postponed.generator([=] { checkPostponed(history, id); }),
|
|
||||||
postponed.type
|
|
||||||
});
|
|
||||||
if (base::take(state->thenRequestEntry)) {
|
|
||||||
requestDialogEntry(history);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
Assert(action == Action::Postpone);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
checkEmptyState(history);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Histories::Action Histories::chooseAction(
|
void Histories::cancelRequest(not_null<History*> history, int id) {
|
||||||
State &state,
|
const auto state = lookup(history);
|
||||||
RequestType type,
|
if (!state) {
|
||||||
bool fromPostponed) const {
|
return;
|
||||||
switch (type) {
|
|
||||||
case RequestType::ReadInbox:
|
|
||||||
for (const auto &[_, sent] : state.sent) {
|
|
||||||
if (sent.type == RequestType::ReadInbox
|
|
||||||
|| sent.type == RequestType::DialogsEntry
|
|
||||||
|| sent.type == RequestType::Delete) {
|
|
||||||
if (!fromPostponed) {
|
|
||||||
auto &postponed = state.postponed;
|
|
||||||
for (auto i = begin(postponed); i != end(postponed);) {
|
|
||||||
if (i->second.type == RequestType::ReadInbox) {
|
|
||||||
i = postponed.erase(i);
|
|
||||||
} else {
|
|
||||||
++i;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return Action::Postpone;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return Action::Send;
|
|
||||||
|
|
||||||
case RequestType::DialogsEntry:
|
|
||||||
for (const auto &[_, sent] : state.sent) {
|
|
||||||
if (sent.type == RequestType::DialogsEntry) {
|
|
||||||
return Action::Skip;
|
|
||||||
}
|
|
||||||
if (sent.type == RequestType::ReadInbox
|
|
||||||
|| sent.type == RequestType::Delete) {
|
|
||||||
if (!fromPostponed) {
|
|
||||||
auto &postponed = state.postponed;
|
|
||||||
for (const auto &[_, postponed] : state.postponed) {
|
|
||||||
if (postponed.type == RequestType::DialogsEntry) {
|
|
||||||
return Action::Skip;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return Action::Postpone;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return Action::Send;
|
|
||||||
|
|
||||||
case RequestType::History:
|
|
||||||
for (const auto &[_, sent] : state.sent) {
|
|
||||||
if (sent.type == RequestType::Delete) {
|
|
||||||
return Action::Postpone;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return Action::Send;
|
|
||||||
|
|
||||||
case RequestType::Delete:
|
|
||||||
for (const auto &[_, sent] : state.sent) {
|
|
||||||
if (sent.type == RequestType::History
|
|
||||||
|| sent.type == RequestType::ReadInbox) {
|
|
||||||
return Action::Postpone;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
for (auto i = begin(state.sent); i != end(state.sent);) {
|
|
||||||
if (i->second.type == RequestType::DialogsEntry) {
|
|
||||||
session().api().request(i->second.id).cancel();
|
|
||||||
i = state.sent.erase(i);
|
|
||||||
state.thenRequestEntry = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return Action::Send;
|
|
||||||
}
|
}
|
||||||
Unexpected("Request type in Histories::chooseAction.");
|
state->postponed.remove(id);
|
||||||
|
finishSentRequest(history, state, id);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Histories::finishSentRequest(
|
||||||
|
not_null<History*> history,
|
||||||
|
not_null<State*> state,
|
||||||
|
int id) {
|
||||||
|
state->sent.remove(id);
|
||||||
|
if (!state->postponed.empty() && !postponeHistoryRequest(*state)) {
|
||||||
|
for (auto &[id, postponed] : base::take(state->postponed)) {
|
||||||
|
const auto requestId = postponed.generator([=] {
|
||||||
|
checkPostponed(history, id);
|
||||||
|
});
|
||||||
|
state->sent.emplace(id, SentRequest{
|
||||||
|
std::move(postponed.generator),
|
||||||
|
requestId,
|
||||||
|
RequestType::History
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (state->postponedRequestEntry && !postponeEntryRequest(*state)) {
|
||||||
|
const auto i = _dialogRequests.find(history);
|
||||||
|
Assert(i != end(_dialogRequests));
|
||||||
|
const auto [j, ok] = _dialogRequestsPending.emplace(
|
||||||
|
history,
|
||||||
|
std::move(i->second));
|
||||||
|
Assert(ok);
|
||||||
|
state->postponedRequestEntry = false;
|
||||||
|
}
|
||||||
|
checkEmptyState(history);
|
||||||
}
|
}
|
||||||
|
|
||||||
Histories::State *Histories::lookup(not_null<History*> history) {
|
Histories::State *Histories::lookup(not_null<History*> history) {
|
||||||
|
|
|
@ -23,6 +23,14 @@ class Folder;
|
||||||
|
|
||||||
class Histories final {
|
class Histories final {
|
||||||
public:
|
public:
|
||||||
|
enum class RequestType : uchar {
|
||||||
|
None,
|
||||||
|
History,
|
||||||
|
ReadInbox,
|
||||||
|
Delete,
|
||||||
|
Send,
|
||||||
|
};
|
||||||
|
|
||||||
explicit Histories(not_null<Session*> owner);
|
explicit Histories(not_null<Session*> owner);
|
||||||
|
|
||||||
[[nodiscard]] Session &owner() const;
|
[[nodiscard]] Session &owner() const;
|
||||||
|
@ -49,34 +57,28 @@ public:
|
||||||
void changeDialogUnreadMark(not_null<History*> history, bool unread);
|
void changeDialogUnreadMark(not_null<History*> history, bool unread);
|
||||||
//void changeDialogUnreadMark(not_null<Data::Feed*> feed, bool unread); // #feed
|
//void changeDialogUnreadMark(not_null<Data::Feed*> feed, bool unread); // #feed
|
||||||
|
|
||||||
|
int sendRequest(
|
||||||
|
not_null<History*> history,
|
||||||
|
RequestType type,
|
||||||
|
Fn<mtpRequestId(Fn<void()> done)> generator);
|
||||||
|
void cancelRequest(not_null<History*> history, int id);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
enum class RequestType : uchar {
|
struct PostponedHistoryRequest {
|
||||||
None,
|
|
||||||
DialogsEntry,
|
|
||||||
History,
|
|
||||||
ReadInbox,
|
|
||||||
Delete,
|
|
||||||
};
|
|
||||||
enum class Action : uchar {
|
|
||||||
Send,
|
|
||||||
Postpone,
|
|
||||||
Skip,
|
|
||||||
};
|
|
||||||
struct PostponedRequest {
|
|
||||||
Fn<mtpRequestId(Fn<void()> done)> generator;
|
Fn<mtpRequestId(Fn<void()> done)> generator;
|
||||||
RequestType type = RequestType::None;
|
|
||||||
};
|
};
|
||||||
struct SentRequest {
|
struct SentRequest {
|
||||||
|
Fn<mtpRequestId(Fn<void()> done)> generator;
|
||||||
mtpRequestId id = 0;
|
mtpRequestId id = 0;
|
||||||
RequestType type = RequestType::None;
|
RequestType type = RequestType::None;
|
||||||
};
|
};
|
||||||
struct State {
|
struct State {
|
||||||
base::flat_map<int, PostponedRequest> postponed;
|
base::flat_map<int, PostponedHistoryRequest> postponed;
|
||||||
base::flat_map<int, SentRequest> sent;
|
base::flat_map<int, SentRequest> sent;
|
||||||
crl::time readWhen = 0;
|
crl::time readWhen = 0;
|
||||||
MsgId readTill = 0;
|
MsgId readTill = 0;
|
||||||
int autoincrement = 0;
|
int autoincrement = 0;
|
||||||
bool thenRequestEntry = false;
|
bool postponedRequestEntry = false;
|
||||||
};
|
};
|
||||||
|
|
||||||
void readInboxTill(not_null<History*> history, MsgId tillId, bool force);
|
void readInboxTill(not_null<History*> history, MsgId tillId, bool force);
|
||||||
|
@ -84,15 +86,13 @@ private:
|
||||||
void sendReadRequest(not_null<History*> history, State &state);
|
void sendReadRequest(not_null<History*> history, State &state);
|
||||||
[[nodiscard]] State *lookup(not_null<History*> history);
|
[[nodiscard]] State *lookup(not_null<History*> history);
|
||||||
void checkEmptyState(not_null<History*> history);
|
void checkEmptyState(not_null<History*> history);
|
||||||
int sendRequest(
|
void checkPostponed(not_null<History*> history, int id);
|
||||||
|
void finishSentRequest(
|
||||||
not_null<History*> history,
|
not_null<History*> history,
|
||||||
RequestType type,
|
not_null<State*> state,
|
||||||
Fn<mtpRequestId(Fn<void()> done)> generator);
|
int id);
|
||||||
void checkPostponed(not_null<History*> history, int requestId);
|
[[nodiscard]] bool postponeHistoryRequest(const State &state) const;
|
||||||
[[nodiscard]] Action chooseAction(
|
[[nodiscard]] bool postponeEntryRequest(const State &state) const;
|
||||||
State &state,
|
|
||||||
RequestType type,
|
|
||||||
bool fromPostponed = false) const;
|
|
||||||
|
|
||||||
void sendDialogRequests();
|
void sendDialogRequests();
|
||||||
void applyPeerDialogs(const MTPmessages_PeerDialogs &dialogs);
|
void applyPeerDialogs(const MTPmessages_PeerDialogs &dialogs);
|
||||||
|
|
Loading…
Reference in New Issue