/* This file is part of Telegram Desktop, an unofficial desktop messaging app, see https://telegram.org Telegram Desktop is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. It 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 General Public License for more details. Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE Copyright (c) 2014 John Preston, https://tdesktop.com */ #pragma once #include "mtpConnection.h" #include "mtpDC.h" #include "mtpRPC.h" class MTProtoSession; class MTPSessionData { public: MTPSessionData(MTProtoSession *creator) : _session(0), _salt(0), fakeRequestId(-2000000000) , _messagesSent(0), keyChecked(false) , _owner(creator) { } void setSession(uint64 session) { DEBUG_LOG(("MTP Info: setting server_session: %1").arg(session)); QWriteLocker locker(&lock); if (_session != session) { _session = session; _messagesSent = 0; } } uint64 getSession() const { QReadLocker locker(&lock); return _session; } void setSalt(uint64 salt) { QWriteLocker locker(&lock); _salt = salt; } uint64 getSalt() const { QReadLocker locker(&lock); return _salt; } const mtpAuthKeyPtr &getKey() const { return authKey; } void setKey(const mtpAuthKeyPtr &key) { if (authKey != key) { uint64 session; memset_rand(&session, sizeof(uint64)); authKey = key; DEBUG_LOG(("MTP Info: new auth key set in SessionData, id %1, setting random server_session %2").arg(key ? key->keyId() : 0).arg(session)); setSession(session); } } bool isCheckedKey() const { QReadLocker locker(&lock); return keyChecked; } void setCheckedKey(bool checked) { QWriteLocker locker(&lock); keyChecked = checked; } QReadWriteLock *keyMutex() const; QReadWriteLock *toSendMutex() const { return &toSendLock; } QReadWriteLock *haveSentMutex() const { return &haveSentLock; } QReadWriteLock *toResendMutex() const { return &toResendLock; } QReadWriteLock *wereAckedMutex() const { return &wereAckedLock; } QReadWriteLock *receivedIdsMutex() const { return &receivedIdsLock; } QReadWriteLock *haveReceivedMutex() const { return &haveReceivedLock; } mtpPreRequestMap &toSendMap() { return toSend; } const mtpPreRequestMap &toSendMap() const { return toSend; } mtpRequestMap &haveSentMap() { return haveSent; } const mtpRequestMap &haveSentMap() const { return haveSent; } mtpRequestIdsMap &toResendMap() { // msgId -> requestId, on which toSend: requestId -> request for resended requests return toResend; } const mtpRequestIdsMap &toResendMap() const { return toResend; } mtpMsgIdsSet &receivedIdsSet() { return receivedIds; } const mtpMsgIdsSet &receivedIdsSet() const { return receivedIds; } mtpRequestIdsMap &wereAckedMap() { return wereAcked; } const mtpRequestIdsMap &wereAckedMap() const { return wereAcked; } mtpResponseMap &haveReceivedMap() { return haveReceived; } const mtpResponseMap &haveReceivedMap() const { return haveReceived; } mtpRequestId nextFakeRequestId() { // must be locked by haveReceivedMutex() if (haveReceived.isEmpty() || haveReceived.cbegin().key() > 0) { fakeRequestId = -2000000000; } else { ++fakeRequestId; } return fakeRequestId; } MTProtoSession *owner() { return _owner; } const MTProtoSession *owner() const { return _owner; } uint32 nextRequestSeqNumber(bool needAck = true) { QWriteLocker locker(&lock); uint32 result(_messagesSent); _messagesSent += (needAck ? 1 : 0); return result * 2 + (needAck ? 1 : 0); } void clear(); private: uint64 _session, _salt; uint32 _messagesSent; mtpRequestId fakeRequestId; MTProtoSession *_owner; mtpAuthKeyPtr authKey; bool keyChecked; mtpPreRequestMap toSend; // map of request_id -> request, that is waiting to be sent mtpRequestMap 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 mtpRequestIdsMap toResend; // map of msg_id -> request_id, that request_id -> request lies in toSend and is waiting to be resent mtpMsgIdsSet receivedIds; // set of received msg_id's, for checking new msg_ids mtpRequestIdsMap wereAcked; // map of msg_id -> request_id, this msg_ids already were acked or do not need ack mtpResponseMap haveReceived; // map of request_id -> response, that should be processed in other thread // mutexes mutable QReadWriteLock lock; mutable QReadWriteLock toSendLock; mutable QReadWriteLock haveSentLock; mutable QReadWriteLock toResendLock; mutable QReadWriteLock receivedIdsLock; mutable QReadWriteLock wereAckedLock; mutable QReadWriteLock haveReceivedLock; }; class MTProtoSession : public QObject { Q_OBJECT public: MTProtoSession(); void start(int32 dcenter = 0, uint32 connects = 0); void restart(); void stop(); int32 getDC() const; ~MTProtoSession(); QReadWriteLock *keyMutex() const; void keyCreated(const mtpAuthKeyPtr &key); void destroyKey(); template mtpRequestId send(const TRequest &request, RPCResponseHandler callbacks = RPCResponseHandler(), uint64 msCanWait = 0, uint32 layer = 0, bool toMainDC = false); // send mtp request void cancel(mtpRequestId requestId); int32 requestState(mtpRequestId requestId) const; int32 getState() const; QString transport() const; mtpRequestId resend(mtpMsgId msgId, uint64 msCanWait = 0, bool forceContainer = false, bool sendMsgStateInfo = false); void resendAll(); // after connection restart void sendPrepared(const mtpRequest &request, uint64 msCanWait = 0, bool newRequest = true); // nulls msgId and seqNo in request, if newRequest = true void sendPreparedWithInit(const mtpRequest &request, uint64 msCanWait = 0); signals: void authKeyCreated(); void needToSend(); void needToSendAsync(); // emit this signal, to emit needToSend() in MTProtoSession thread void startSendTimer(int msec); // manipulating timer from all threads void stopSendTimer(); public slots: void authKeyCreatedForDC(); void tryToReceive(); void checkRequestsByTimer(); void onConnectionStateChange(qint32 newState); private: template mtpRequestId sendFirst(const MTPInitConnection &request, RPCResponseHandler callbacks = RPCResponseHandler(), uint64 msCanWait = 0, uint32 layer = 0, bool toMainDC = false); // send first mtp request typedef QList MTProtoConnections; MTProtoConnections connections; MTPSessionData data; int32 dcId; MTProtoDCPtr dc; uint64 msSendCall, msWait; QTimer timeouter; QTimer sender; }; inline QReadWriteLock *MTPSessionData::keyMutex() const { return _owner->keyMutex(); } MTPrpcError rpcClientError(const QString &type, const QString &description = QString()); // here typedef QSharedPointer MTProtoSessionPtr;