tdesktop/Telegram/SourceFiles/mtproto/mtpSession.h

273 lines
7.0 KiB
C++

/*
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 <typename TRequest>
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 <typename TRequest>
mtpRequestId sendFirst(const MTPInitConnection<TRequest> &request, RPCResponseHandler callbacks = RPCResponseHandler(), uint64 msCanWait = 0, uint32 layer = 0, bool toMainDC = false); // send first mtp request
typedef QList<MTProtoConnection*> 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<MTProtoSession> MTProtoSessionPtr;