/*
Copyright (C) 2012-23 Miranda NG team (https://miranda-ng.org)
This program 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 version 2
of the License.
This program 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.
You should have received a copy of the GNU General Public License
along with this program. If not, see .
*/
#include "stdafx.h"
void CTelegramProto::OnEndSession(td::ClientManager::Response&)
{
m_bTerminated = true;
}
void __cdecl CTelegramProto::ServerThread(void *)
{
m_bTerminated = m_bAuthorized = false;
m_szFullPhone.Format("%d%S", (int)m_iCountry, (wchar_t *)m_szOwnPhone);
m_pClientManager = std::make_unique();
m_iClientId = m_pClientManager->create_client_id();
SendQuery(new TD::getOption("version"));
while (!m_bTerminated) {
ProcessResponse(m_pClientManager->receive(1));
}
m_pClientManager = std::move(nullptr);
}
void CTelegramProto::LogOut()
{
if (m_bTerminated)
return;
debugLogA("CTelegramProto::OnLoggedOut");
m_bTerminated = true;
m_bAuthorized = false;
ProtoBroadcastAck(0, ACKTYPE_STATUS, ACKRESULT_SUCCESS, (HANDLE)m_iStatus, ID_STATUS_OFFLINE);
m_iStatus = m_iDesiredStatus = ID_STATUS_OFFLINE;
m_impl.m_keepAlive.Stop();
setAllContactStatuses(ID_STATUS_OFFLINE, false);
for (auto &it : m_arUsers)
it->m_si = nullptr;
}
void CTelegramProto::OnLoggedIn()
{
m_bAuthorized = true;
debugLogA("CTelegramProto::OnLoggedIn");
ProtoBroadcastAck(0, ACKTYPE_STATUS, ACKRESULT_SUCCESS, (HANDLE)m_iStatus, m_iDesiredStatus);
m_iStatus = m_iDesiredStatus;
if (m_bUnregister) {
SendQuery(new TD::terminateSession());
SendQuery(new TD::logOut(), &CTelegramProto::OnEndSession);
}
else {
m_impl.m_keepAlive.Start(1000);
SendQuery(new TD::getChats(td::tl::unique_ptr(), 1000));
}
}
///////////////////////////////////////////////////////////////////////////////
void CTelegramProto::SendKeepAlive()
{
time_t now = time(0);
for (auto &it : m_arUsers) {
if (it->m_timer1 && now - it->m_timer1 > STATUS_SWITCH_TIMEOUT) {
it->m_timer1 = 0;
it->m_timer2 = now;
setWord(it->hContact, "Status", ID_STATUS_NA);
}
else if (it->m_timer2 && now - it->m_timer2 > STATUS_SWITCH_TIMEOUT) {
it->m_timer2 = 0;
setWord(it->hContact, "Status", ID_STATUS_OFFLINE);
}
}
}
void CTelegramProto::SendDeleteMsg()
{
m_impl.m_deleteMsg.Stop();
mir_cslock lck(m_csDeleteMsg);
SendQuery(new TD::deleteMessages(m_deleteChatId, std::move(m_deleteIds), true));
m_deleteChatId = 0;
}
void CTelegramProto::SendMarkRead()
{
m_impl.m_markRead.Stop();
mir_cslock lck(m_csMarkRead);
SendQuery(new TD::viewMessages(m_markChatId, 0, std::move(m_markIds), true));
m_markChatId = 0;
}
///////////////////////////////////////////////////////////////////////////////
void CTelegramProto::ProcessResponse(td::ClientManager::Response response)
{
if (!response.object)
return;
debugLogA("ProcessResponse: id=%d (%s)", int(response.request_id), to_string(response.object).c_str());
if (response.request_id) {
TG_REQUEST tmp(response.request_id, 0);
auto *p = m_arRequests.find(&tmp);
if (p) {
p->Execute(this, response);
m_arRequests.remove(p);
}
return;
}
switch (response.object->get_id()) {
case TD::updateAuthorizationState::ID:
ProcessAuth((TD::updateAuthorizationState *)response.object.get());
break;
case TD::updateBasicGroup::ID:
ProcessBasicGroup((TD::updateBasicGroup*)response.object.get());
break;
case TD::updateChatFilters::ID:
ProcessGroups((TD::updateChatFilters *)response.object.get());
break;
case TD::updateChatLastMessage::ID:
ProcessChatLastMessage((TD::updateChatLastMessage *)response.object.get());
break;
case TD::updateChatNotificationSettings::ID:
ProcessChatNotification((TD::updateChatNotificationSettings*)response.object.get());
break;
case TD::updateChatPosition::ID:
ProcessChatPosition((TD::updateChatPosition *)response.object.get());
break;
case TD::updateChatReadInbox::ID:
ProcessMarkRead((TD::updateChatReadInbox *)response.object.get());
break;
case TD::updateDeleteMessages::ID:
ProcessDeleteMessage((TD::updateDeleteMessages*)response.object.get());
break;
case TD::updateConnectionState::ID:
ProcessConnectionState((TD::updateConnectionState *)response.object.get());
break;
case TD::updateFile::ID:
ProcessFile((TD::updateFile *)response.object.get());
break;
case TD::updateNewChat::ID:
ProcessChat((TD::updateNewChat *)response.object.get());
break;
case TD::updateNewMessage::ID:
ProcessMessage((TD::updateNewMessage *)response.object.get());
break;
case TD::updateSupergroup::ID:
ProcessSuperGroup((TD::updateSupergroup *)response.object.get());
break;
case TD::updateUserStatus::ID:
ProcessStatus((TD::updateUserStatus *)response.object.get());
break;
case TD::updateUser::ID:
ProcessUser((TD::updateUser *)response.object.get());
break;
}
}
/////////////////////////////////////////////////////////////////////////////////////////
void CTelegramProto::OnSendMessage(td::ClientManager::Response &response, void *pUserInfo)
{
if (!response.object)
return;
if (response.object->get_id() != TD::message::ID) {
debugLogA("Gotten class ID %d instead of %d, exiting", response.object->get_id(), TD::message::ID);
return;
}
auto *pMessage = ((TD::message *)response.object.get());
auto *pUser = FindChat(pMessage->chat_id_);
if (pUser) {
char szMsgId[100];
_i64toa(pMessage->id_, szMsgId, 10);
ProtoBroadcastAck(pUser->hContact, ACKTYPE_MESSAGE, ACKRESULT_SUCCESS, pUserInfo, (LPARAM)szMsgId);
}
}
int CTelegramProto::SendTextMessage(int64_t chatId, const char *pszMessage)
{
int ret = m_iMsgId++;
auto pContent = TD::make_object();
pContent->text_ = TD::make_object();
pContent->text_->text_ = std::move(pszMessage);
auto *pMessage = new TD::sendMessage();
pMessage->chat_id_ = chatId;
pMessage->input_message_content_ = std::move(pContent);
SendQuery(pMessage, &CTelegramProto::OnSendMessage, (void *)ret);
return ret;
}
void CTelegramProto::SendQuery(TD::Function *pFunc, TG_QUERY_HANDLER pHandler)
{
if (!m_pClientManager)
return;
int queryId = ++m_iQueryId;
auto szDescr = to_string(*pFunc);
debugLogA("Sending query %d:\n%s", queryId, szDescr.c_str());
m_pClientManager->send(m_iClientId, queryId, TD::object_ptr(pFunc));
if (pHandler)
m_arRequests.insert(new TG_REQUEST(queryId, pHandler));
}
void CTelegramProto::SendQuery(TD::Function *pFunc, TG_QUERY_HANDLER_FULL pHandler, void *pUserInfo)
{
if (!m_pClientManager)
return;
int queryId = ++m_iQueryId;
auto szDescr = to_string(*pFunc);
debugLogA("Sending full query %d:\n%s", queryId, szDescr.c_str());
m_pClientManager->send(m_iClientId, queryId, TD::object_ptr(pFunc));
if (pHandler)
m_arRequests.insert(new TG_REQUEST_FULL(queryId, pHandler, pUserInfo));
}
///////////////////////////////////////////////////////////////////////////////
void CTelegramProto::ProcessBasicGroup(TD::updateBasicGroup *pObj)
{
auto *pBasicGroup = pObj->basic_group_.get();
if (pBasicGroup->upgraded_to_supergroup_id_)
if (auto *pUser = FindUser(pBasicGroup->upgraded_to_supergroup_id_)) {
pUser->bLoadMembers = true;
if (pUser->m_si)
pUser->m_si->bHasNicklist = true;
}
auto iStatusId = pBasicGroup->status_->get_id();
if (iStatusId == TD::chatMemberStatusBanned::ID) {
debugLogA("We are banned here, skipping");
return;
}
TG_BASIC_GROUP tmp(pBasicGroup->id_, 0);
auto *pGroup = m_arBasicGroups.find(&tmp);
if (pGroup == nullptr) {
pGroup = new TG_BASIC_GROUP(tmp.id, std::move(pObj->basic_group_));
m_arBasicGroups.insert(pGroup);
}
else pGroup->group = std::move(pObj->basic_group_);
TG_USER *pUser;
if (iStatusId == TD::chatMemberStatusLeft::ID) {
pUser = AddFakeUser(tmp.id, true);
pUser->wszLastName.Format(TranslateT("%d member(s)"), pGroup->group->member_count_);
}
else pUser = AddUser(tmp.id, true);
pUser->bLoadMembers = true;
}
void CTelegramProto::ProcessChat(TD::updateNewChat *pObj)
{
int64_t chatId;
auto *pChat = pObj->chat_.get();
std::string szTitle;
switch(pChat->type_->get_id()) {
case TD::chatTypePrivate::ID:
case TD::chatTypeSecret::ID:
chatId = pChat->id_;
break;
case TD::chatTypeBasicGroup::ID:
chatId = ((TD::chatTypeBasicGroup*)pChat->type_.get())->basic_group_id_;
szTitle = pChat->title_;
break;
case TD::chatTypeSupergroup::ID:
chatId = ((TD::chatTypeSupergroup *)pChat->type_.get())->supergroup_id_;
szTitle = pChat->title_;
break;
default:
debugLogA("Invalid chat type %d, ignoring", pChat->type_->get_id());
return;
}
if (auto *pUser = FindUser(chatId)) {
pUser->chatId = pChat->id_;
if (!m_arChats.find(pUser))
m_arChats.insert(pUser);
if (!szTitle.empty()) {
if (pUser->hContact != INVALID_CONTACT_ID)
setUString(pUser->hContact, "Nick", szTitle.c_str());
else if (pUser->wszNick.IsEmpty())
pUser->wszFirstName = Utf2T(szTitle.c_str());
}
if (CheckSearchUser(pUser))
return;
if (pUser->isGroupChat && pUser->hContact != INVALID_CONTACT_ID)
InitGroupChat(pUser, pChat);
}
else debugLogA("Unknown chat id %lld, ignoring", chatId);
}
void CTelegramProto::ProcessChatLastMessage(TD::updateChatLastMessage *pObj)
{
auto *pUser = FindChat(pObj->chat_id_);
if (pUser == nullptr) {
debugLogA("Unknown chat, skipping");
return;
}
if (pUser->hContact == INVALID_CONTACT_ID) {
debugLogA("Last message for a temporary contact, skipping");
return;
}
// according to #3406 we wipe history for the contacts from contacts' list
// but remove the contact itself if it's a temporary one
if (pObj->last_message_ == nullptr) {
if (Contact::OnList(pUser->hContact))
CallService(MS_HISTORY_EMPTY, pUser->hContact, TRUE);
else
db_delete_contact(pUser->hContact, true);
}
}
void CTelegramProto::ProcessChatNotification(TD::updateChatNotificationSettings *pObj)
{
auto *pUser = FindChat(pObj->chat_id_);
if (pUser == nullptr || pUser->hContact == INVALID_CONTACT_ID)
return;
auto &pSettings = pObj->notification_settings_;
if (!pSettings->use_default_mute_for_ && pSettings->mute_for_ != 0)
Chat_Mute(pUser->hContact, CHATMODE_MUTE);
else
Chat_Mute(pUser->hContact, CHATMODE_NORMAL);
memcpy(&pUser->notificationSettings, pSettings.get(), sizeof(pUser->notificationSettings));
}
void CTelegramProto::ProcessChatPosition(TD::updateChatPosition *pObj)
{
if (pObj->position_->get_id() != TD::chatPosition::ID) {
debugLogA("Unsupport position");
return;
}
auto *pUser = FindChat(pObj->chat_id_);
if (pUser == nullptr) {
debugLogA("Unknown chat, skipping");
return;
}
if (pUser->hContact == INVALID_CONTACT_ID)
return;
auto *pPos = (TD::chatPosition *)pObj->position_.get();
if (pPos->list_) {
auto *pList = (TD::chatListFilter *)pPos->list_.get();
CMStringA szSetting(FORMAT, "ChatFilter%d", pList->chat_filter_id_);
CMStringW wszGroup(getMStringW(szSetting));
if (!wszGroup.IsEmpty()) {
ptrW pwszExistingGroup(Clist_GetGroup(pUser->hContact));
if (!pwszExistingGroup
|| (!pUser->isGroupChat && !mir_wstrcmp(pwszExistingGroup, m_wszDefaultGroup))
|| (pUser->isGroupChat && !mir_wstrcmp(pwszExistingGroup, ptrW(Chat_GetGroup())))) {
CMStringW wszNewGroup(FORMAT, L"%s\\%s", (wchar_t *)m_wszDefaultGroup, wszGroup.c_str());
Clist_GroupCreate(0, wszNewGroup);
Clist_SetGroup(pUser->hContact, wszNewGroup);
}
}
}
}
void CTelegramProto::ProcessConnectionState(TD::updateConnectionState *pObj)
{
pConnState = std::move(pObj->state_);
switch (pConnState->get_id()) {
case TD::connectionStateConnecting::ID:
debugLogA("Connection state: connecting");
break;
case TD::connectionStateConnectingToProxy::ID:
debugLogA("Connection state: connecting to proxy");
break;
case TD::connectionStateWaitingForNetwork::ID:
debugLogA("Connection state: waiting for network");
break;
case TD::connectionStateUpdating::ID:
debugLogA("Connection state: updating");
break;
case TD::connectionStateReady::ID:
debugLogA("Connection state: connected");
if (pAuthState->get_id() == TD::authorizationStateReady::ID)
OnLoggedIn();
break;
}
}
void CTelegramProto::ProcessDeleteMessage(TD::updateDeleteMessages *pObj)
{
if (!pObj->is_permanent_)
return;
auto *pUser = FindChat(pObj->chat_id_);
if (pUser == nullptr || pUser->hContact == INVALID_CONTACT_ID) {
debugLogA("message from unknown chat, ignored");
return;
}
for (auto &it : pObj->message_ids_) {
char id[100];
_i64toa(it, id, 10);
if (MEVENT hEvent = db_event_getById(m_szModuleName, id))
db_event_delete(hEvent, true);
}
}
void CTelegramProto::ProcessGroups(TD::updateChatFilters *pObj)
{
for (auto &grp : pObj->chat_filters_) {
if (grp->icon_name_ != "Custom")
continue;
CMStringA szSetting(FORMAT, "ChatFilter%d", grp->id_);
CMStringW wszOldValue(getMStringW(szSetting));
Utf2T wszNewValue(grp->title_.c_str());
if (wszOldValue.IsEmpty()) {
Clist_GroupCreate(m_iBaseGroup, wszNewValue);
setWString(szSetting, wszNewValue);
}
else if (wszOldValue != wszNewValue) {
CMStringW wszFullGroup(FORMAT, L"%s\\%s", (wchar_t *)m_wszDefaultGroup, wszNewValue.get());
MGROUP oldGroup = Clist_GroupExists(wszFullGroup);
if (!oldGroup)
Clist_GroupCreate(m_iBaseGroup, wszFullGroup);
else
Clist_GroupRename(oldGroup, wszFullGroup);
setWString(szSetting, wszNewValue);
}
}
}
void CTelegramProto::ProcessMarkRead(TD::updateChatReadInbox *pObj)
{
auto *pUser = FindChat(pObj->chat_id_);
if (pUser == nullptr) {
debugLogA("message from unknown chat/user, ignored");
return;
}
char szId[100];
_i64toa(pObj->last_read_inbox_message_id_, szId, 10);
MEVENT hLastRead = db_event_getById(m_szModuleName, szId);
if (hLastRead == 0) {
debugLogA("unknown event, ignored");
return;
}
bool bExit = false;
for (MEVENT hEvent = db_event_firstUnread(pUser->hContact); hEvent; hEvent = db_event_next(pUser->hContact, hEvent)) {
if (bExit)
break;
bExit = (hEvent == hLastRead);
DBEVENTINFO dbei = {};
if (db_event_get(hEvent, &dbei))
continue;
if (!dbei.markedRead())
db_event_markRead(pUser->hContact, hEvent, true);
}
}
void CTelegramProto::ProcessMessage(TD::updateNewMessage *pObj)
{
auto *pMessage = pObj->message_.get();
auto *pUser = FindChat(pMessage->chat_id_);
if (pUser == nullptr) {
debugLogA("message from unknown chat/user, ignored");
return;
}
CMStringA szText(GetMessageText(pUser, pMessage));
if (szText.IsEmpty()) {
debugLogA("this message was not processed, ignored");
return;
}
// make a temporary contact if needed
if (pUser->hContact == INVALID_CONTACT_ID) {
if (pUser->isGroupChat) {
debugLogA("spam from unknown group chat, ignored");
return;
}
AddUser(pUser->id, false);
Contact::RemoveFromList(pUser->hContact);
}
char szId[100], szUserId[100];
_i64toa(pMessage->id_, szId, 10);
PROTORECVEVENT pre = {};
pre.szMessage = szText.GetBuffer();
pre.szMsgId = szId;
pre.timestamp = pMessage->date_;
if (pMessage->is_outgoing_)
pre.flags |= PREF_SENT;
if (GetGcUserId(pUser, pMessage, szUserId))
pre.szUserId = szUserId;
ProtoChainRecvMsg(pUser->hContact, &pre);
}
void CTelegramProto::ProcessStatus(TD::updateUserStatus *pObj)
{
if (auto *pUser = FindUser(pObj->user_id_)) {
if (pUser->hContact == INVALID_CONTACT_ID)
return;
switch (pObj->status_->get_id()) {
case TD::userStatusOnline::ID:
setWord(pUser->hContact, "Status", ID_STATUS_ONLINE);
break;
case TD::userStatusRecently::ID:
case TD::userStatusOffline::ID:
setWord(pUser->hContact, "Status", ID_STATUS_AWAY);
pUser->m_timer1 = time(0);
break;
default:
debugLogA("!!!!! Unknown status packet, report it to the developers");
}
}
}
void CTelegramProto::ProcessSuperGroup(TD::updateSupergroup *pObj)
{
auto iStatusId = pObj->supergroup_->status_->get_id();
if (iStatusId == TD::chatMemberStatusBanned::ID) {
debugLogA("We are banned here, skipping");
return;
}
TG_SUPER_GROUP tmp(pObj->supergroup_->id_, 0);
auto *pGroup = m_arSuperGroups.find(&tmp);
if (pGroup == nullptr) {
pGroup = new TG_SUPER_GROUP(tmp.id, std::move(pObj->supergroup_));
m_arSuperGroups.insert(pGroup);
}
else pGroup->group = std::move(pObj->supergroup_);
if (iStatusId == TD::chatMemberStatusLeft::ID) {
auto *pUser = AddFakeUser(tmp.id, true);
pUser->wszNick = getName(pGroup->group->usernames_.get());
pUser->wszLastName.Format(TranslateT("%d member(s)"), pGroup->group->member_count_);
}
else AddUser(tmp.id, true);
}
void CTelegramProto::ProcessUser(TD::updateUser *pObj)
{
auto *pUser = pObj->user_.get();
if (pUser->phone_number_ == m_szFullPhone.c_str()) {
m_iOwnId = pUser->id_;
SetId(0, m_iOwnId);
if (!FindUser(pUser->id_)) {
auto *pMe = new TG_USER(pUser->id_, 0);
m_arUsers.insert(pMe);
m_arChats.insert(pMe);
}
}
else if (!pUser->is_contact_) {
auto *pu = AddFakeUser(pUser->id_, false);
if (pu->hContact != INVALID_CONTACT_ID)
Contact::RemoveFromList(pu->hContact);
pu->wszFirstName = Utf2T(pUser->first_name_.c_str());
pu->wszLastName = Utf2T(pUser->last_name_.c_str());
if (pUser->usernames_) {
pu->wszNick = L"@";
pu->wszNick.Append(Utf2T(pUser->usernames_->editable_username_.c_str()));
}
else {
pu->wszNick = Utf2T(pUser->first_name_.c_str());
if (!pUser->last_name_.empty())
pu->wszNick.AppendFormat(L" %s", Utf2T(pUser->last_name_.c_str()).get());
}
CheckSearchUser(pu);
debugLogA("User doesn't belong to your contacts, skipping");
return;
}
auto *pu = AddUser(pUser->id_, false);
UpdateString(pu->hContact, "FirstName", pUser->first_name_);
UpdateString(pu->hContact, "LastName", pUser->last_name_);
UpdateString(pu->hContact, "Phone", pUser->phone_number_);
if (pUser->usernames_)
UpdateString(pu->hContact, "Nick", pUser->usernames_->editable_username_);
if (pu->hContact == 0)
pu->wszNick = Contact::GetInfo(CNF_DISPLAY, 0, m_szModuleName);
Contact::PutOnList(pu->hContact);
if (pUser->is_premium_)
ExtraIcon_SetIconByName(g_plugin.m_hIcon, pu->hContact, "tg_premium");
else
ExtraIcon_SetIconByName(g_plugin.m_hIcon, pu->hContact, nullptr);
if (auto *pPhoto = pUser->profile_photo_.get()) {
if (auto *pSmall = pPhoto->small_.get()) {
auto remoteId = pSmall->remote_->unique_id_;
auto storedId = getMStringA(pu->hContact, DBKEY_AVATAR_HASH);
if (remoteId != storedId.c_str()) {
if (!remoteId.empty()) {
pu->szAvatarHash = remoteId.c_str();
setString(pu->hContact, DBKEY_AVATAR_HASH, remoteId.c_str());
SendQuery(new TD::downloadFile(pSmall->id_, 5, 0, 0, false));
}
else delSetting(pu->hContact, DBKEY_AVATAR_HASH);
}
}
}
if (pUser->status_) {
if (pUser->status_->get_id() == TD::userStatusOffline::ID) {
auto *pOffline = (TD::userStatusOffline *)pUser->status_.get();
setDword(pu->hContact, "LastSeen", pOffline->was_online_);
}
}
}