198 lines
7.5 KiB
C++
198 lines
7.5 KiB
C++
/*
|
|
* This is the source code of tgnet library v. 1.0
|
|
* It is licensed under GNU GPL v. 2 or later.
|
|
* You should have received a copy of the license in this archive (see LICENSE).
|
|
*
|
|
* Copyright Nikolai Kudashov, 2015.
|
|
*/
|
|
|
|
#ifndef CONNECTIONSMANAGER_H
|
|
#define CONNECTIONSMANAGER_H
|
|
|
|
#include <pthread.h>
|
|
#include <queue>
|
|
#include <functional>
|
|
#include <sys/epoll.h>
|
|
#include <map>
|
|
#include <atomic>
|
|
#include <bits/unique_ptr.h>
|
|
#include "Defines.h"
|
|
|
|
#ifdef ANDROID
|
|
#include <jni.h>
|
|
#endif
|
|
|
|
class NativeByteBuffer;
|
|
class Connection;
|
|
class Datacenter;
|
|
class Request;
|
|
class DatacenterHandshake;
|
|
class TLObject;
|
|
class ConnectionSocket;
|
|
class TL_auth_exportedAuthorization;
|
|
class ByteArray;
|
|
class TL_config;
|
|
class EventObject;
|
|
class Config;
|
|
|
|
class ConnectionsManager {
|
|
|
|
public:
|
|
ConnectionsManager();
|
|
~ConnectionsManager();
|
|
|
|
static ConnectionsManager &getInstance();
|
|
int64_t getCurrentTimeMillis();
|
|
int32_t getCurrentTime();
|
|
int32_t getTimeDifference();
|
|
int32_t sendRequest(TLObject *object, onCompleteFunc onComplete, onQuickAckFunc onQuickAck, uint32_t flags, uint32_t datacenterId, ConnectionType connetionType, bool immediate);
|
|
void sendRequest(TLObject *object, onCompleteFunc onComplete, onQuickAckFunc onQuickAck, uint32_t flags, uint32_t datacenterId, ConnectionType connetionType, bool immediate, int32_t requestToken);
|
|
void cancelRequest(int32_t token, bool notifyServer);
|
|
void cleanUp();
|
|
void cancelRequestsForGuid(int32_t guid);
|
|
void bindRequestToGuid(int32_t requestToken, int32_t guid);
|
|
void applyDatacenterAddress(uint32_t datacenterId, std::string ipAddress, uint32_t port);
|
|
void setDelegate(ConnectiosManagerDelegate *connectiosManagerDelegate);
|
|
ConnectionState getConnectionState();
|
|
void setUserId(int32_t userId);
|
|
void switchBackend();
|
|
void resumeNetwork(bool partial);
|
|
void pauseNetwork();
|
|
void setNetworkAvailable(bool value);
|
|
void setUseIpv6(bool value);
|
|
void init(uint32_t version, int32_t layer, int32_t apiId, std::string deviceModel, std::string systemVersion, std::string appVersion, std::string langCode, std::string configPath, std::string logPath, int32_t userId, bool isPaused, bool enablePushConnection);
|
|
void updateDcSettings(uint32_t datacenterId);
|
|
void setPushConnectionEnabled(bool value);
|
|
|
|
#ifdef ANDROID
|
|
void sendRequest(TLObject *object, onCompleteFunc onComplete, onQuickAckFunc onQuickAck, uint32_t flags, uint32_t datacenterId, ConnectionType connetionType, bool immediate, int32_t requestToken, jobject ptr1, jobject ptr2);
|
|
static void useJavaVM(JavaVM *vm, bool useJavaByteBuffers);
|
|
#endif
|
|
|
|
private:
|
|
static void *ThreadProc(void *data);
|
|
|
|
void initDatacenters();
|
|
void loadConfig();
|
|
void saveConfig();
|
|
void select();
|
|
void wakeup();
|
|
void processServerResponse(TLObject *message, int64_t messageId, int32_t messageSeqNo, int64_t messageSalt, Connection *connection, int64_t innerMsgId, int64_t containerMessageId);
|
|
void sendPing(Datacenter *datacenter, bool usePushConnection);
|
|
void sendMessagesToConnection(std::vector<std::unique_ptr<NetworkMessage>> &messages, Connection *connection, bool reportAck);
|
|
void sendMessagesToConnectionWithConfirmation(std::vector<std::unique_ptr<NetworkMessage>> &messages, Connection *connection, bool reportAck);
|
|
void requestSaltsForDatacenter(Datacenter *datacenter);
|
|
void clearRequestsForDatacenter(Datacenter *datacenter);
|
|
void registerForInternalPushUpdates();
|
|
void processRequestQueue(uint32_t connectionType, uint32_t datacenterId);
|
|
void moveToDatacenter(uint32_t datacenterId);
|
|
void authorizeOnMovingDatacenter();
|
|
void authorizedOnMovingDatacenter();
|
|
Datacenter *getDatacenterWithId(uint32_t datacenterId);
|
|
std::unique_ptr<TLObject> wrapInLayer(TLObject *object, Datacenter *datacenter, Request *baseRequest);
|
|
void removeRequestFromGuid(int32_t requestToken);
|
|
void cancelRequestInternal(int32_t token, bool notifyServer, bool removeFromClass);
|
|
int callEvents(int64_t now);
|
|
|
|
void checkPendingTasks();
|
|
void scheduleTask(std::function<void()> task);
|
|
void scheduleEvent(EventObject *eventObject, uint32_t time);
|
|
void removeEvent(EventObject *eventObject);
|
|
void onConnectionClosed(Connection *connection);
|
|
void onConnectionConnected(Connection *connection);
|
|
void onConnectionQuickAckReceived(Connection *connection, int32_t ack);
|
|
void onConnectionDataReceived(Connection *connection, NativeByteBuffer *data, uint32_t length);
|
|
void attachConnection(ConnectionSocket *connection);
|
|
void detachConnection(ConnectionSocket *connection);
|
|
TLObject *TLdeserialize(TLObject *request, uint32_t bytes, NativeByteBuffer *data);
|
|
TLObject *getRequestWithMessageId(int64_t messageId);
|
|
void onDatacenterHandshakeComplete(Datacenter *datacenter, int32_t timeDiff);
|
|
void onDatacenterExportAuthorizationComplete(Datacenter *datacenter);
|
|
int64_t generateMessageId();
|
|
bool isIpv6Enabled();
|
|
bool isNetworkAvailable();
|
|
|
|
uint32_t configVersion = 2;
|
|
Config *config = nullptr;
|
|
|
|
std::list<EventObject *> events;
|
|
|
|
std::map<uint32_t, Datacenter *> datacenters;
|
|
std::map<int32_t, std::vector<std::int32_t>> quickAckIdToRequestIds;
|
|
int32_t pingTime;
|
|
bool testBackend = false;
|
|
std::atomic<uint32_t> lastRequestToken{1};
|
|
uint32_t currentDatacenterId = 0;
|
|
uint32_t movingToDatacenterId = DEFAULT_DATACENTER_ID;
|
|
int64_t pushSessionId = 0;
|
|
int32_t currentPingTime = 0;
|
|
bool registeringForPush = false;
|
|
int64_t lastPushPingTime = 0;
|
|
bool sendingPushPing = false;
|
|
bool updatingDcSettings = false;
|
|
int32_t updatingDcStartTime = 0;
|
|
int32_t lastDcUpdateTime = 0;
|
|
int64_t lastPingTime = getCurrentTimeMillis();
|
|
bool networkPaused = false;
|
|
int32_t nextSleepTimeout = CONNECTION_BACKGROUND_KEEP_TIME;
|
|
int64_t lastPauseTime = 0;
|
|
ConnectionState connectionState = ConnectionStateConnecting;
|
|
std::unique_ptr<ByteArray> movingAuthorization;
|
|
std::vector<int64_t> sessionsToDestroy;
|
|
int32_t lastDestroySessionRequestTime;
|
|
std::map<int32_t, std::vector<int32_t>> requestsByGuids;
|
|
std::map<int32_t, int32_t> guidsByRequests;
|
|
|
|
pthread_t networkThread;
|
|
pthread_mutex_t mutex;
|
|
std::queue<std::function<void()>> pendingTasks;
|
|
struct epoll_event *epollEvents;
|
|
timespec timeSpec;
|
|
int32_t timeDifference = 0;
|
|
int64_t lastOutgoingMessageId = 0;
|
|
bool networkAvailable = true;
|
|
bool ipv6Enabled = false;
|
|
std::vector<ConnectionSocket *> activeConnections;
|
|
int epolFd;
|
|
int *pipeFd;
|
|
NativeByteBuffer *networkBuffer;
|
|
|
|
requestsList requestsQueue;
|
|
requestsList runningRequests;
|
|
std::vector<uint32_t> requestingSaltsForDc;
|
|
int32_t lastPingId = 0;
|
|
|
|
uint32_t currentVersion = 1;
|
|
int32_t currentLayer = 34;
|
|
int32_t currentApiId = 6;
|
|
std::string currentDeviceModel;
|
|
std::string currentSystemVersion;
|
|
std::string currentAppVersion;
|
|
std::string currentLangCode;
|
|
std::string currentConfigPath;
|
|
std::string currentLogPath;
|
|
int32_t currentUserId = 0;
|
|
bool registeredForInternalPush = false;
|
|
bool pushConnectionEnabled = true;
|
|
|
|
ConnectiosManagerDelegate *delegate;
|
|
|
|
friend class ConnectionSocket;
|
|
friend class ConnectionSession;
|
|
friend class Connection;
|
|
friend class Timer;
|
|
friend class Datacenter;
|
|
friend class TL_message;
|
|
friend class TL_rpc_result;
|
|
friend class Config;
|
|
};
|
|
|
|
#ifdef ANDROID
|
|
extern JavaVM *javaVm;
|
|
extern JNIEnv *jniEnv;
|
|
extern jclass jclass_ByteBuffer;
|
|
extern jmethodID jclass_ByteBuffer_allocateDirect;
|
|
#endif
|
|
|
|
#endif
|