mirror of https://github.com/procxx/kepka.git
439 lines
11 KiB
C++
439 lines
11 KiB
C++
/*
|
|
This file is part of Telegram Desktop,
|
|
the official desktop application for the Telegram messaging service.
|
|
|
|
For license and copyright information please follow this link:
|
|
https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|
*/
|
|
#pragma once
|
|
|
|
#include "base/timer.h"
|
|
#include "mtproto/mtproto_rpc_sender.h"
|
|
#include "mtproto/mtproto_proxy_data.h"
|
|
|
|
#include <QtCore/QTimer>
|
|
|
|
namespace MTP {
|
|
|
|
class Instance;
|
|
class AuthKey;
|
|
using AuthKeyPtr = std::shared_ptr<AuthKey>;
|
|
|
|
namespace internal {
|
|
|
|
// Received msgIds and wereAcked msgIds count stored.
|
|
constexpr auto kIdsBufferSize = 400;
|
|
|
|
class Dcenter;
|
|
class Connection;
|
|
|
|
using PreRequestMap = QMap<mtpRequestId, SecureRequest>;
|
|
using RequestMap = QMap<mtpMsgId, SecureRequest>;
|
|
|
|
class RequestIdsMap : public QMap<mtpMsgId, mtpRequestId> {
|
|
public:
|
|
using ParentType = QMap<mtpMsgId, mtpRequestId>;
|
|
|
|
mtpMsgId min() const {
|
|
return size() ? cbegin().key() : 0;
|
|
}
|
|
|
|
mtpMsgId max() const {
|
|
ParentType::const_iterator e(cend());
|
|
return size() ? (--e).key() : 0;
|
|
}
|
|
|
|
};
|
|
|
|
class ReceivedMsgIds {
|
|
public:
|
|
bool registerMsgId(mtpMsgId msgId, bool needAck) {
|
|
auto i = _idsNeedAck.constFind(msgId);
|
|
if (i == _idsNeedAck.cend()) {
|
|
if (_idsNeedAck.size() < kIdsBufferSize || msgId > min()) {
|
|
_idsNeedAck.insert(msgId, needAck);
|
|
return true;
|
|
}
|
|
MTP_LOG(-1, ("No need to handle - %1 < min = %2").arg(msgId).arg(min()));
|
|
} else {
|
|
MTP_LOG(-1, ("No need to handle - %1 already is in map").arg(msgId));
|
|
}
|
|
return false;
|
|
}
|
|
|
|
mtpMsgId min() const {
|
|
return _idsNeedAck.isEmpty() ? 0 : _idsNeedAck.cbegin().key();
|
|
}
|
|
|
|
mtpMsgId max() const {
|
|
auto end = _idsNeedAck.cend();
|
|
return _idsNeedAck.isEmpty() ? 0 : (--end).key();
|
|
}
|
|
|
|
void shrink() {
|
|
auto size = _idsNeedAck.size();
|
|
while (size-- > kIdsBufferSize) {
|
|
_idsNeedAck.erase(_idsNeedAck.begin());
|
|
}
|
|
}
|
|
|
|
enum class State {
|
|
NotFound,
|
|
NeedsAck,
|
|
NoAckNeeded,
|
|
};
|
|
State lookup(mtpMsgId msgId) const {
|
|
auto i = _idsNeedAck.constFind(msgId);
|
|
if (i == _idsNeedAck.cend()) {
|
|
return State::NotFound;
|
|
}
|
|
return i.value() ? State::NeedsAck : State::NoAckNeeded;
|
|
}
|
|
|
|
void clear() {
|
|
_idsNeedAck.clear();
|
|
}
|
|
|
|
private:
|
|
QMap<mtpMsgId, bool> _idsNeedAck;
|
|
|
|
};
|
|
|
|
using SerializedMessage = mtpBuffer;
|
|
|
|
inline bool ResponseNeedsAck(const SerializedMessage &response) {
|
|
if (response.size() < 8) {
|
|
return false;
|
|
}
|
|
auto seqNo = *(uint32*)(response.constData() + 6);
|
|
return (seqNo & 0x01) ? true : false;
|
|
}
|
|
|
|
struct ConnectionOptions {
|
|
ConnectionOptions() = default;
|
|
ConnectionOptions(
|
|
const QString &systemLangCode,
|
|
const QString &cloudLangCode,
|
|
const QString &langPackName,
|
|
const ProxyData &proxy,
|
|
bool useIPv4,
|
|
bool useIPv6,
|
|
bool useHttp,
|
|
bool useTcp);
|
|
ConnectionOptions(const ConnectionOptions &other) = default;
|
|
ConnectionOptions &operator=(const ConnectionOptions &other) = default;
|
|
|
|
QString systemLangCode;
|
|
QString cloudLangCode;
|
|
QString langPackName;
|
|
ProxyData proxy;
|
|
bool useIPv4 = true;
|
|
bool useIPv6 = true;
|
|
bool useHttp = true;
|
|
bool useTcp = true;
|
|
|
|
};
|
|
|
|
class Session;
|
|
class SessionData {
|
|
public:
|
|
SessionData(not_null<Session*> creator) : _owner(creator) {
|
|
}
|
|
|
|
void setCurrentKeyId(uint64 keyId);
|
|
void setSessionId(uint64 sessionId) {
|
|
DEBUG_LOG(("MTP Info: setting server_session: %1").arg(sessionId));
|
|
|
|
QWriteLocker locker(&_lock);
|
|
if (_sessionId != sessionId) {
|
|
_sessionId = sessionId;
|
|
_messagesSent = 0;
|
|
}
|
|
}
|
|
[[nodiscard]] uint64 getSessionId() const {
|
|
QReadLocker locker(&_lock);
|
|
return _sessionId;
|
|
}
|
|
void notifyConnectionInited(const ConnectionOptions &options);
|
|
void setConnectionOptions(ConnectionOptions options) {
|
|
QWriteLocker locker(&_lock);
|
|
_options = options;
|
|
}
|
|
[[nodiscard]] ConnectionOptions connectionOptions() const {
|
|
QReadLocker locker(&_lock);
|
|
return _options;
|
|
}
|
|
|
|
void setSalt(uint64 salt) {
|
|
QWriteLocker locker(&_lock);
|
|
_salt = salt;
|
|
}
|
|
[[nodiscard]] uint64 getSalt() const {
|
|
QReadLocker locker(&_lock);
|
|
return _salt;
|
|
}
|
|
|
|
[[nodiscard]] const AuthKeyPtr &getKeyForCheck() const {
|
|
return _dcKeyForCheck;
|
|
}
|
|
void setKeyForCheck(const AuthKeyPtr &key);
|
|
|
|
not_null<QReadWriteLock*> toSendMutex() const {
|
|
return &_toSendLock;
|
|
}
|
|
not_null<QReadWriteLock*> haveSentMutex() const {
|
|
return &_haveSentLock;
|
|
}
|
|
not_null<QReadWriteLock*> toResendMutex() const {
|
|
return &_toResendLock;
|
|
}
|
|
not_null<QReadWriteLock*> wereAckedMutex() const {
|
|
return &_wereAckedLock;
|
|
}
|
|
not_null<QReadWriteLock*> receivedIdsMutex() const {
|
|
return &_receivedIdsLock;
|
|
}
|
|
not_null<QReadWriteLock*> haveReceivedMutex() const {
|
|
return &_haveReceivedLock;
|
|
}
|
|
not_null<QReadWriteLock*> stateRequestMutex() const {
|
|
return &_stateRequestLock;
|
|
}
|
|
|
|
PreRequestMap &toSendMap() {
|
|
return _toSend;
|
|
}
|
|
const PreRequestMap &toSendMap() const {
|
|
return _toSend;
|
|
}
|
|
RequestMap &haveSentMap() {
|
|
return _haveSent;
|
|
}
|
|
const RequestMap &haveSentMap() const {
|
|
return _haveSent;
|
|
}
|
|
RequestIdsMap &toResendMap() { // msgId -> requestId, on which toSend: requestId -> request for resended requests
|
|
return _toResend;
|
|
}
|
|
const RequestIdsMap &toResendMap() const {
|
|
return _toResend;
|
|
}
|
|
ReceivedMsgIds &receivedIdsSet() {
|
|
return _receivedIds;
|
|
}
|
|
const ReceivedMsgIds &receivedIdsSet() const {
|
|
return _receivedIds;
|
|
}
|
|
RequestIdsMap &wereAckedMap() {
|
|
return _wereAcked;
|
|
}
|
|
const RequestIdsMap &wereAckedMap() const {
|
|
return _wereAcked;
|
|
}
|
|
QMap<mtpRequestId, SerializedMessage> &haveReceivedResponses() {
|
|
return _receivedResponses;
|
|
}
|
|
const QMap<mtpRequestId, SerializedMessage> &haveReceivedResponses() const {
|
|
return _receivedResponses;
|
|
}
|
|
QList<SerializedMessage> &haveReceivedUpdates() {
|
|
return _receivedUpdates;
|
|
}
|
|
const QList<SerializedMessage> &haveReceivedUpdates() const {
|
|
return _receivedUpdates;
|
|
}
|
|
QMap<mtpMsgId, bool> &stateRequestMap() {
|
|
return _stateRequest;
|
|
}
|
|
const QMap<mtpMsgId, bool> &stateRequestMap() const {
|
|
return _stateRequest;
|
|
}
|
|
|
|
// Warning! Valid only in constructor, _owner is guaranteed != null.
|
|
[[nodiscard]] not_null<Session*> owner() {
|
|
return _owner;
|
|
}
|
|
|
|
uint32 nextRequestSeqNumber(bool needAck = true) {
|
|
QWriteLocker locker(&_lock);
|
|
auto result = _messagesSent;
|
|
_messagesSent += (needAck ? 1 : 0);
|
|
return result * 2 + (needAck ? 1 : 0);
|
|
}
|
|
|
|
void clearForNewKey(not_null<Instance*> instance);
|
|
|
|
// Connection -> Session interface.
|
|
void queueTryToReceive();
|
|
void queueNeedToResumeAndSend();
|
|
void queueConnectionStateChange(int newState);
|
|
void queueResendAll();
|
|
void queueResetDone();
|
|
void queueSendAnything(crl::time msCanWait = 0);
|
|
void queueSendMsgsStateInfo(quint64 msgId, QByteArray data);
|
|
void queueResend(
|
|
mtpMsgId msgId,
|
|
crl::time msCanWait,
|
|
bool forceContainer,
|
|
bool sendMsgStateInfo);
|
|
void queueResendMany(
|
|
QVector<mtpMsgId> msgIds,
|
|
crl::time msCanWait,
|
|
bool forceContainer,
|
|
bool sendMsgStateInfo);
|
|
|
|
[[nodiscard]] bool connectionInited() const;
|
|
[[nodiscard]] AuthKeyPtr getPersistentKey() const;
|
|
[[nodiscard]] AuthKeyPtr getTemporaryKey() const;
|
|
[[nodiscard]] bool acquireKeyCreation();
|
|
void releaseKeyCreationOnDone(
|
|
const AuthKeyPtr &temporaryKey,
|
|
const AuthKeyPtr &persistentKey);
|
|
void releaseKeyCreationOnFail();
|
|
void destroyTemporaryKey(uint64 keyId);
|
|
|
|
void detach();
|
|
|
|
private:
|
|
template <typename Callback>
|
|
void withSession(Callback &&callback);
|
|
|
|
uint64 _keyId = 0;
|
|
uint64 _sessionId = 0;
|
|
uint64 _salt = 0;
|
|
uint32 _messagesSent = 0;
|
|
|
|
Session *_owner = nullptr;
|
|
mutable QMutex _ownerMutex;
|
|
|
|
AuthKeyPtr _dcKeyForCheck;
|
|
ConnectionOptions _options;
|
|
|
|
PreRequestMap _toSend; // map of request_id -> request, that is waiting to be sent
|
|
RequestMap _haveSent; // map of msg_id -> request, that was sent, msDate = 0 for msgs_state_req (no resend / state req), msDate = 0, seqNo = 0 for containers
|
|
RequestIdsMap _toResend; // map of msg_id -> request_id, that request_id -> request lies in toSend and is waiting to be resent
|
|
ReceivedMsgIds _receivedIds; // set of received msg_id's, for checking new msg_ids
|
|
RequestIdsMap _wereAcked; // map of msg_id -> request_id, this msg_ids already were acked or do not need ack
|
|
QMap<mtpMsgId, bool> _stateRequest; // set of msg_id's, whose state should be requested
|
|
|
|
QMap<mtpRequestId, SerializedMessage> _receivedResponses; // map of request_id -> response that should be processed in the main thread
|
|
QList<SerializedMessage> _receivedUpdates; // list of updates that should be processed in the main thread
|
|
|
|
// mutexes
|
|
mutable QReadWriteLock _lock;
|
|
mutable QReadWriteLock _toSendLock;
|
|
mutable QReadWriteLock _haveSentLock;
|
|
mutable QReadWriteLock _toResendLock;
|
|
mutable QReadWriteLock _receivedIdsLock;
|
|
mutable QReadWriteLock _wereAckedLock;
|
|
mutable QReadWriteLock _haveReceivedLock;
|
|
mutable QReadWriteLock _stateRequestLock;
|
|
|
|
};
|
|
|
|
class Session : public QObject {
|
|
Q_OBJECT
|
|
|
|
public:
|
|
// Main thread.
|
|
Session(
|
|
not_null<Instance*> instance,
|
|
ShiftedDcId shiftedDcId,
|
|
Dcenter *dc);
|
|
~Session();
|
|
|
|
void start();
|
|
void reInitConnection();
|
|
|
|
void restart();
|
|
void refreshOptions();
|
|
void stop();
|
|
void kill();
|
|
|
|
void unpaused();
|
|
|
|
// Thread-safe.
|
|
[[nodiscard]] ShiftedDcId getDcWithShift() const;
|
|
[[nodiscard]] AuthKeyPtr getPersistentKey() const;
|
|
[[nodiscard]] AuthKeyPtr getTemporaryKey() const;
|
|
[[nodiscard]] bool connectionInited() const;
|
|
|
|
// Connection thread.
|
|
[[nodiscard]] bool acquireKeyCreation();
|
|
void releaseKeyCreationOnDone(
|
|
const AuthKeyPtr &temporaryKey,
|
|
const AuthKeyPtr &persistentKey);
|
|
void releaseKeyCreationOnFail();
|
|
void destroyTemporaryKey(uint64 keyId);
|
|
|
|
void notifyDcConnectionInited();
|
|
|
|
void ping();
|
|
void cancel(mtpRequestId requestId, mtpMsgId msgId);
|
|
int32 requestState(mtpRequestId requestId) const;
|
|
int32 getState() const;
|
|
QString transport() const;
|
|
|
|
void sendDcKeyCheck(const AuthKeyPtr &key);
|
|
|
|
// Nulls msgId and seqNo in request, if newRequest = true.
|
|
void sendPrepared(
|
|
const SecureRequest &request,
|
|
crl::time msCanWait = 0,
|
|
bool newRequest = true);
|
|
|
|
void tryToReceive();
|
|
void needToResumeAndSend();
|
|
void connectionStateChange(int newState);
|
|
void resendAll(); // After connection restart.
|
|
void resetDone();
|
|
void sendAnything(crl::time msCanWait = 0);
|
|
void sendMsgsStateInfo(quint64 msgId, QByteArray data);
|
|
mtpRequestId resend(
|
|
mtpMsgId msgId,
|
|
crl::time msCanWait = 0,
|
|
bool forceContainer = false,
|
|
bool sendMsgStateInfo = false);
|
|
|
|
signals:
|
|
void authKeyChanged();
|
|
void needToSend();
|
|
void needToPing();
|
|
void needToRestart();
|
|
|
|
private:
|
|
[[nodiscard]] bool sharedDc() const;
|
|
void checkRequestsByTimer();
|
|
void watchDcKeyChanges();
|
|
|
|
bool rpcErrorOccured(mtpRequestId requestId, const RPCFailHandlerPtr &onFail, const RPCError &err);
|
|
|
|
const not_null<Instance*> _instance;
|
|
const ShiftedDcId _shiftedDcId = 0;
|
|
const std::unique_ptr<Dcenter> _ownedDc;
|
|
const not_null<Dcenter*> _dc;
|
|
const std::shared_ptr<SessionData> _data;
|
|
|
|
std::unique_ptr<Connection> _connection;
|
|
|
|
bool _killed = false;
|
|
bool _needToReceive = false;
|
|
|
|
AuthKeyPtr _dcKeyForCheck;
|
|
bool _myKeyCreation = false;
|
|
|
|
crl::time _msSendCall = 0;
|
|
crl::time _msWait = 0;
|
|
|
|
bool _ping = false;
|
|
|
|
base::Timer _timeouter;
|
|
base::Timer _sender;
|
|
|
|
rpl::lifetime _lifetime;
|
|
|
|
};
|
|
|
|
} // namespace internal
|
|
} // namespace MTP
|