/* Copyright (C) 2012-24 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_botIds.clear(); m_bTerminated = m_bAuthorized = false; m_pClientManager = std::make_unique(); m_iClientId = m_pClientManager->create_client_id(); SendQuery(new TD::getOption("version")); NETLIBUSERSETTINGS nluSettings; Netlib_GetUserSettings(m_hNetlibUser, &nluSettings); if (nluSettings.useProxy) { TD::object_ptr proxyType; switch (nluSettings.proxyType) { case PROXYTYPE_SOCKS4: case PROXYTYPE_SOCKS5: proxyType = TD::make_object(); break; case PROXYTYPE_HTTP: case PROXYTYPE_HTTPS: proxyType = TD::make_object(); break; } if (proxyType) SendQuery(new TD::addProxy(nluSettings.szProxyServer, nluSettings.wProxyPort, true, std::move(proxyType))); } while (!m_bTerminated) { ProcessResponse(m_pClientManager->receive(1)); } m_pClientManager = std::move(nullptr); } void CTelegramProto::UnregisterSession() { if (getByte(DBKEY_AUTHORIZED)) { m_bUnregister = true; if (m_pClientManager) { SendQuery(new TD::terminateSession()); SendQuery(new TD::logOut(), &CTelegramProto::OnEndSession); } else ForkThread(&CTelegramProto::ServerThread); } } 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; setWord(m_iSavedMessages, "Status", ID_STATUS_OFFLINE); m_impl.m_keepAlive.Stop(); setAllContactStatuses(ID_STATUS_OFFLINE, false); for (auto &it : m_arUsers) it->m_si = nullptr; } /////////////////////////////////////////////////////////////////////////////// void CTelegramProto::OnGetChats(td::ClientManager::Response &response) { if (!response.object) return; if (response.object->get_id() != TD::chats::ID) { debugLogA("Gotten class ID %d instead of %d, exiting", response.object->get_id(), TD::chats::ID); return; } auto *pChats = (TD::chats *)response.object.get(); for (auto &it : pChats->chat_ids_) { if (auto *pUser = FindChat(it)) Contact::PutOnList(pUser->hContact); else SendQuery(new TD::getChat(it)); } } void CTelegramProto::OnLoggedIn() { m_bAuthorized = true; debugLogA("CTelegramProto::OnLoggedIn"); ProtoBroadcastAck(0, ACKTYPE_STATUS, ACKRESULT_SUCCESS, (HANDLE)m_iStatus, m_iDesiredStatus); m_iStatus = m_iDesiredStatus; setWord(m_iSavedMessages, "Status", ID_STATUS_ONLINE); if (m_bUnregister) { SendQuery(new TD::terminateSession()); SendQuery(new TD::logOut(), &CTelegramProto::OnEndSession); } else { m_impl.m_keepAlive.Start(1000); setByte(DBKEY_AUTHORIZED, 1); SendQuery(new TD::getChats(td::tl::unique_ptr(), 1000), &CTelegramProto::OnGetChats); } } /////////////////////////////////////////////////////////////////////////////// void CTelegramProto::SendKeepAlive() { time_t now = time(0); int iDiff1 = m_iTimeDiff1, iDiff2 = m_iTimeDiff2; for (auto &it : m_arUsers) { if (it->m_timer1 && now - it->m_timer1 > iDiff1) { it->m_timer1 = 0; // if the second status is set in the options, enable the second timer if (m_iTimeDiff2) { it->m_timer2 = now; setWord(it->hContact, "Status", m_iStatus2); } else setWord(it->hContact, "Status", ID_STATUS_OFFLINE); } else if (it->m_timer2 && now - it->m_timer2 > iDiff2) { 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), m_bDeleteForAll)); m_deleteChatId = 0; } void CTelegramProto::SendMarkRead() { m_impl.m_markRead.Stop(); mir_cslock lck(m_csMarkRead); SendQuery(new TD::viewMessages(m_markChatId, std::move(m_markIds), 0, 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); mir_cslock lck(m_csRequests); auto *p = m_arRequests.find(&tmp); if (p) { p->Execute(this, response); m_arRequests.remove(p); } return; } switch (response.object->get_id()) { case TD::updateActiveEmojiReactions::ID: ProcessActiveEmoji((TD::updateActiveEmojiReactions *)response.object.get()); break; case TD::updateAuthorizationState::ID: ProcessAuth((TD::updateAuthorizationState *)response.object.get()); break; case TD::updateBasicGroup::ID: ProcessBasicGroup((TD::updateBasicGroup*)response.object.get()); break; case TD::updateBasicGroupFullInfo::ID: ProcessBasicGroupInfo((TD::updateBasicGroupFullInfo *)response.object.get()); break; case TD::updateChatAction::ID: ProcessChatAction((TD::updateChatAction *)response.object.get()); break; case TD::updateChatAvailableReactions::ID: ProcessChatReactions((TD::updateChatAvailableReactions *)response.object.get()); break; case TD::updateChatFolders::ID: ProcessGroups((TD::updateChatFolders *)response.object.get()); break; case TD::updateChatHasProtectedContent::ID: ProcessChatHasProtected((TD::updateChatHasProtectedContent *)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::updateForumTopicInfo::ID: ProcessForum((TD::updateForumTopicInfo*)response.object.get()); break; case TD::updateMessageContent::ID: ProcessMessageContent((TD::updateMessageContent *)response.object.get()); break; case TD::updateMessageInteractionInfo::ID: ProcessMessageReactions((TD::updateMessageInteractionInfo *)response.object.get()); break; case TD::updateMessageSendSucceeded::ID: { auto *pUpdate = (TD::updateMessageSendSucceeded *)response.object.get(); auto *pMessage = pUpdate->message_.get(); auto szOldId = msg2id(pMessage->chat_id_, pUpdate->old_message_id_); if (pUpdate->old_message_id_) if (auto hDbEvent = db_event_getById(m_szModuleName, szOldId)) db_event_updateId(hDbEvent, msg2id(pMessage)); ProcessMessage(pMessage); TG_OWN_MESSAGE tmp(0, 0, szOldId); if (auto *pOwnMsg = m_arOwnMsg.find(&tmp)) { auto szMsgId = msg2id(pMessage); if (pOwnMsg->hAck) ProtoBroadcastAck(pOwnMsg->hContact ? pOwnMsg->hContact : m_iSavedMessages, ACKTYPE_MESSAGE, ACKRESULT_SUCCESS, pOwnMsg->hAck, (LPARAM)szMsgId.c_str()); if (auto hDbEvent = db_event_getById(m_szModuleName, szMsgId)) db_event_delivered(pOwnMsg->hContact, hDbEvent); m_arOwnMsg.remove(pOwnMsg); } } break; case TD::updateNewChat::ID: ProcessChat((TD::updateNewChat *)response.object.get()); break; case TD::updateNewMessage::ID: { auto *pMessage = ((TD::updateNewMessage *)response.object.get())->message_.get(); TG_OWN_MESSAGE tmp(0, 0, msg2id(pMessage)); if (!m_arOwnMsg.find(&tmp)) ProcessMessage(pMessage); } break; case TD::updateOption::ID: ProcessOption((TD::updateOption *)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) { if (!response.object) return; switch (response.object->get_id()) { case TD::error::ID: for (auto &it : m_arOwnMsg) if (it->hAck == (HANDLE)response.request_id) { auto *pError = ((TD::error *)response.object.get()); CMStringW wszMsg(FORMAT, TranslateT("Error %d: %s"), pError->code_, TranslateW(Utf2T(pError->message_.c_str()))); ProtoBroadcastAck(it->hContact, ACKTYPE_MESSAGE, ACKRESULT_FAILED, it->hAck, (LPARAM)wszMsg.c_str()); break; } break; case TD::message::ID: for (auto &it : m_arOwnMsg) if (it->hAck == (HANDLE)response.request_id) { auto *pMessage = ((TD::message *)response.object.get()); it->szMsgId = msg2id(pMessage); break; } break; default: debugLogA("Gotten class ID %d instead of %d, exiting", response.object->get_id(), TD::message::ID); return; } } int CTelegramProto::SendTextMessage(int64_t chatId, int64_t threadId, int64_t replyId, const char *pszMessage) { auto pContent = TD::make_object(); pContent->text_ = formatBbcodes(pszMessage); auto *pMessage = new TD::sendMessage(); pMessage->chat_id_ = chatId; pMessage->input_message_content_ = std::move(pContent); pMessage->message_thread_id_ = threadId; pMessage->reply_to_message_id_ = replyId; return SendQuery(pMessage, &CTelegramProto::OnSendMessage); } int CTelegramProto::SendQuery(TD::Function *pFunc, TG_QUERY_HANDLER pHandler) { if (!m_pClientManager) return -1; 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) { mir_cslock lck(m_csRequests); m_arRequests.insert(new TG_REQUEST(queryId, pHandler)); } return queryId; } int CTelegramProto::SendQuery(TD::Function *pFunc, TG_QUERY_HANDLER_FULL pHandler, void *pUserInfo) { if (!m_pClientManager) return -1; 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) { mir_cslock lck(m_csRequests); m_arRequests.insert(new TG_REQUEST_FULL(queryId, pHandler, pUserInfo)); } return queryId; } /////////////////////////////////////////////////////////////////////////////// void CTelegramProto::OnGetHistory(td::ClientManager::Response &response, void *pUserInfo) { if (!response.object) return; if (response.object->get_id() != TD::messages::ID) { debugLogA("Gotten class ID %d instead of %d, exiting", response.object->get_id(), TD::messages::ID); return; } auto *pUser = (TG_USER *)pUserInfo; TD::int53 lastMsgId = INT64_MAX; auto *pMessages = (TD::messages *)response.object.get(); for (auto &it : pMessages->messages_) { auto *pMsg = it.get(); if (pMsg->id_ < lastMsgId) lastMsgId = pMsg->id_; char szUserId[100]; auto szMsgId(msg2id(pMsg)); if (db_event_getById(m_szModuleName, szMsgId)) continue; CMStringA szBody = GetMessageText(pUser, pMsg, true, true), szReplyId; if (szBody.IsEmpty()) continue; DBEVENTINFO dbei = {}; dbei.eventType = EVENTTYPE_MESSAGE; dbei.szModule = m_szModuleName; dbei.timestamp = pMsg->date_; dbei.cbBlob = szBody.GetLength(); dbei.pBlob = szBody.GetBuffer(); dbei.szId = szMsgId; dbei.flags = DBEF_READ | DBEF_UTF; if (pMsg->is_outgoing_) dbei.flags |= DBEF_SENT; if (this->GetGcUserId(pUser, pMsg, szUserId)) dbei.szUserId = szUserId; if (pMsg->reply_to_message_id_) { szReplyId = msg2id(pMsg->chat_id_, pMsg->reply_to_message_id_); dbei.szReplyId = szReplyId; } db_event_add(GetRealContact(pUser), &dbei); } pUser->nHistoryChunks--; if (pUser->isForum) { if (lastMsgId != INT64_MAX && pUser->nHistoryChunks > 0) SendQuery(new TD::getMessageThreadHistory(pUser->chatId, lastMsgId, lastMsgId, 0, 100), &CTelegramProto::OnGetHistory, pUser); else delete pUser; } else if (lastMsgId != INT64_MAX && pUser->nHistoryChunks > 0) SendQuery(new TD::getChatHistory(pUser->chatId, lastMsgId, 0, 100, false), &CTelegramProto::OnGetHistory, pUser); } INT_PTR CTelegramProto::SvcLoadServerHistory(WPARAM hContact, LPARAM) { TD::int53 lastMsgId = 0; if (MEVENT hEvent = db_event_first(hContact)) { DB::EventInfo dbei(hEvent, false); lastMsgId = dbei2id(dbei); } if (TD::int53 threadId = GetId(hContact, DBKEY_THREAD)) { auto chatId = GetId(hContact); auto *pUser = new TG_USER(-1, hContact, true); pUser->chatId = chatId; pUser->isForum = pUser->isGroupChat = true; pUser->nHistoryChunks = 5; SendQuery(new TD::getMessageThreadHistory(chatId, lastMsgId, lastMsgId, 0, 100), &CTelegramProto::OnGetHistory, pUser); } else if (auto *pUser = FindUser(GetId(hContact))) { pUser->nHistoryChunks = 5; SendQuery(new TD::getChatHistory(pUser->chatId, lastMsgId, 0, 100, false), &CTelegramProto::OnGetHistory, pUser); } return 0; } /////////////////////////////////////////////////////////////////////////////// INT_PTR CTelegramProto::SvcEmptyServerHistory(WPARAM hContact, LPARAM lParam) { debugLogW(L"SvcEmptyServerHistory was called for cc=%d (%s)", (int)hContact, Clist_GetContactDisplayName(hContact)); if (auto *pUser = FindUser(GetId(hContact))) if (pUser->chatId != -1) SendQuery(new TD::deleteChatHistory(pUser->chatId, false, (lParam & CDF_FOR_EVERYONE) != 0)); return 0; } /////////////////////////////////////////////////////////////////////////////// void CTelegramProto::ProcessChat(TD::updateNewChat *pObj) { int64_t userId; auto *pChat = pObj->chat_.get(); std::string szTitle; switch (pChat->type_->get_id()) { case TD::chatTypePrivate::ID: case TD::chatTypeSecret::ID: userId = pChat->id_; break; case TD::chatTypeBasicGroup::ID: userId = ((TD::chatTypeBasicGroup *)pChat->type_.get())->basic_group_id_; szTitle = pChat->title_; break; case TD::chatTypeSupergroup::ID: userId = ((TD::chatTypeSupergroup *)pChat->type_.get())->supergroup_id_; szTitle = pChat->title_; break; default: debugLogA("Invalid chat type %d, ignoring", pChat->type_->get_id()); return; } auto *pUser = FindUser(userId); if (pUser == nullptr) { debugLogA("Unknown user id %lld for chat %lld, ignoring", userId, pChat->id_); return; } pUser->chatId = pChat->id_; MCONTACT hContact = (pUser->id == m_iOwnId) ? 0 : pUser->hContact; if (!m_arChats.find(pUser)) m_arChats.insert(pUser); if (!szTitle.empty()) { if (hContact != INVALID_CONTACT_ID) { setUString(hContact, "Nick", szTitle.c_str()); pUser->wszNick = Utf2T(szTitle.c_str()); } else if (pUser->wszNick.IsEmpty()) pUser->wszFirstName = Utf2T(szTitle.c_str()); } if (auto *pPhoto = pChat->photo_.get()) ProcessAvatar(pPhoto->small_.get(), pUser); if (CheckSearchUser(pUser)) return; if (pUser->hContact != INVALID_CONTACT_ID) { if (pChat->has_protected_content_) setByte(pUser->hContact, "Protected", 1); else delSetting(pUser->hContact, "Protected"); if (pChat->permissions_) Contact::Readonly(hContact, !pChat->permissions_->can_send_basic_messages_); if (pUser->isGroupChat && pUser->m_si == nullptr) InitGroupChat(pUser, (pUser->isForum) ? TranslateT("General") : Utf2T(pChat->title_.c_str())); } } void CTelegramProto::ProcessChatAction(TD::updateChatAction *pObj) { auto *pChat = FindChat(pObj->chat_id_); if (pChat == nullptr) { debugLogA("Unknown chat, skipping"); return; } if (pChat->hContact == INVALID_CONTACT_ID) { debugLogA("Last message for a temporary contact, skipping"); return; } switch (pObj->action_->get_id()) { case TD::chatActionTyping::ID: CallService(MS_PROTO_CONTACTISTYPING, pChat->hContact, 1); break; } } void CTelegramProto::ProcessChatHasProtected(TD::updateChatHasProtectedContent *pObj) { if (auto *pChat = FindChat(pObj->chat_id_)) { if (pObj->has_protected_content_) setByte(pChat->hContact, "Protected", 1); else delSetting(pChat->hContact, "Protected"); } } void CTelegramProto::ProcessChatLastMessage(TD::updateChatLastMessage *pObj) { auto *pUser = FindChat(pObj->chat_id_); if (pUser == nullptr) { debugLogA("Unknown chat, skipping"); return; } pUser->bInited = true; if (pUser->hContact == INVALID_CONTACT_ID) { debugLogA("Last message for a temporary contact, skipping"); return; } } 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) { debugLogA("Temporary contact, skipping"); return; } auto *pPos = (TD::chatPosition *)pObj->position_.get(); if (auto *pList = pPos->list_.get()) { CMStringW wszGroup; switch (auto typeId = pList->get_id()) { case TD::chatListArchive::ID: wszGroup = TranslateT("Archive"); break; case TD::chatListMain::ID: // leave group empty if (pUser->folderId != -1) return; break; case TD::chatListFolder::ID: { int iFolderId = ((TD::chatListFolder *)pList)->chat_folder_id_; CMStringA szSetting(FORMAT, "ChatFilter%d", iFolderId); wszGroup = getMStringW(szSetting); if (wszGroup.IsEmpty()) { debugLogA("Empty group name for group #%d, ignored", iFolderId); return; } if (wszGroup == "Unread") return; pUser->folderId = iFolderId; } break; default: debugLogA("Unknown position type ID %d, ignored", typeId); return; } MCONTACT hContact = GetRealContact(pUser); ptrW pwszExistingGroup(Clist_GetGroup(hContact)); debugLogW(L"Existing contact group <%s>, calculated <%s>", pwszExistingGroup.get(), wszGroup.c_str()); wchar_t *pwszDefaultGroup = m_wszDefaultGroup; if (!pwszExistingGroup || pUser->isForum || !mir_wstrncmp(pwszExistingGroup, pwszDefaultGroup, mir_wstrlen(pwszDefaultGroup)) || (pUser->isGroupChat && !mir_wstrcmp(pwszExistingGroup, ptrW(Chat_GetGroup())))) { CMStringW wszNewGroup(pwszDefaultGroup); if (!wszGroup.IsEmpty()) wszNewGroup.AppendFormat(L"\\%s", wszGroup.c_str()); if (pUser->isForum) wszNewGroup.AppendFormat(L"\\%s", pUser->wszNick.c_str()); debugLogW(L"Setting group for %d to %s", hContact, wszNewGroup.c_str()); Clist_GroupCreate(0, wszNewGroup); Clist_SetGroup(hContact, wszNewGroup); } } } void CTelegramProto::ProcessChatReactions(TD::updateChatAvailableReactions *pObj) { if (pObj->available_reactions_->get_id() != TD::chatAvailableReactionsSome::ID) { debugLogA("Unsupported reactions type: %d", pObj->available_reactions_->get_id()); return; } auto &pReactions = ((TD::chatAvailableReactionsSome *)pObj->available_reactions_.get())->reactions_; if (auto *pChat = FindChat(pObj->chat_id_)) { if (!pChat->pReactions) pChat->pReactions = new OBJLIST(1); else pChat->pReactions->destroy(); for (auto &it : pReactions) { if (it->get_id() != TD::reactionTypeEmoji::ID) continue; auto *pEmoji = (TD::reactionTypeEmoji *)it.get(); auto &str = pEmoji->emoji_; pChat->pReactions->insert(mir_strcpy(new char[str.length() + 1], str.c_str())); } if (pChat->pReactions->getCount() == 0) { delete pChat->pReactions; pChat->pReactions = nullptr; } } } 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::ProcessActiveEmoji(TD::updateActiveEmojiReactions *pObj) { m_defaultEmoji.Empty(); for (auto &it : pObj->emojis_) m_defaultEmoji.AppendFormat("%s ", it.c_str()); m_defaultEmoji.TrimRight(); } 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_) if (MEVENT hEvent = db_event_getById(m_szModuleName, msg2id(pObj->chat_id_, it))) db_event_delete(hEvent, CDF_FROM_SERVER); } void CTelegramProto::ProcessGroups(TD::updateChatFolders *pObj) { for (auto &grp : pObj->chat_folders_) { 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; } if (pObj->last_read_inbox_message_id_) pUser->bInited = true; MEVENT hLastRead = db_event_getById(m_szModuleName, msg2id(pObj->chat_id_, pObj->last_read_inbox_message_id_)); 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(const TD::message *pMessage) { auto *pUser = FindChat(pMessage->chat_id_); if (pUser == nullptr) { debugLogA("message from unknown chat/user, ignored"); return; } if (pMessage->sending_state_) if (pMessage->sending_state_->get_id() == TD::messageSendingStatePending::ID) return; char szUserId[100]; auto szMsgId(msg2id(pMessage)); MEVENT hOldEvent = db_event_getById(m_szModuleName, szMsgId); CMStringA szText(GetMessageText(pUser, pMessage)), szReplyId; 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); } MCONTACT hContact = GetRealContact(pUser); if (pMessage->message_thread_id_) { wchar_t buf[100]; mir_snwprintf(buf, L"%lld_%lld", pMessage->chat_id_, pMessage->message_thread_id_); if (auto *si = Chat_Find(buf, m_szModuleName)) hContact = si->hContact; } DB::EventInfo dbei(hOldEvent); dbei.szId = szMsgId; dbei.cbBlob = szText.GetLength(); dbei.timestamp = pMessage->date_; if (pMessage->is_outgoing_) dbei.flags |= DBEF_SENT; if (!pUser->bInited) dbei.flags |= DBEF_READ; if (GetGcUserId(pUser, pMessage, szUserId)) dbei.szUserId = szUserId; if (pMessage->reply_to_message_id_) { szReplyId = msg2id(pMessage->chat_id_, pMessage->reply_to_message_id_); dbei.szReplyId = szReplyId; } if (dbei) { replaceStr(dbei.pBlob, szText.Detach()); db_event_edit(hOldEvent, &dbei, true); } else { dbei.pBlob = szText.GetBuffer(); ProtoChainRecvMsg(hContact, dbei); } } void CTelegramProto::ProcessMessageContent(TD::updateMessageContent *pObj) { auto *pUser = FindChat(pObj->chat_id_); if (pUser == nullptr) { debugLogA("message from unknown chat/user, ignored"); return; } auto szMsgId = msg2id(pObj->chat_id_, pObj->message_id_); MEVENT hDbEvent = db_event_getById(m_szModuleName, szMsgId); if (hDbEvent == 0) { debugLogA("Unknown message with id=%lld (chat id %lld, ignored", pObj->message_id_, pObj->chat_id_); return; } auto msg = TD::make_object(); msg->id_ = pObj->message_id_; msg->sender_id_ = TD::make_object(pObj->chat_id_); msg->content_ = std::move(pObj->new_content_); TG_OWN_MESSAGE tmp(0, 0, szMsgId); if (auto *pOwnMsg = m_arOwnMsg.find(&tmp)) msg->is_outgoing_ = true; CMStringA szText(GetMessageText(pUser, msg.get())); if (szText.IsEmpty()) { debugLogA("this message was not processed, ignored"); return; } DBEVENTINFO dbei = {}; if (db_event_get(hDbEvent, &dbei)) return; dbei.cbBlob = szText.GetLength(); dbei.pBlob = szText.GetBuffer(); db_event_edit(hDbEvent, &dbei, true); } void CTelegramProto::ProcessMessageReactions(TD::updateMessageInteractionInfo *pObj) { auto *pUser = FindChat(pObj->chat_id_); if (pUser == nullptr) { debugLogA("message from unknown chat/user, ignored"); return; } CMStringA szMsgId(msg2id(pObj->chat_id_, pObj->message_id_)); DB::EventInfo dbei(db_event_getById(m_szModuleName, szMsgId)); if (!dbei) { debugLogA("Unknown message with id=%lld (chat id %lld, ignored", pObj->message_id_, pObj->chat_id_); return; } JSONNode reactions; reactions.set_name("r"); if (pObj->interaction_info_) { for (auto &it : pObj->interaction_info_->reactions_) { if (it->type_->get_id() != TD::reactionTypeEmoji::ID) continue; auto *pEmoji = (TD::reactionTypeEmoji *)it->type_.get(); reactions << INT_PARAM(pEmoji->emoji_.c_str(), it->total_count_); } } auto &json = dbei.setJson(); auto it = json.find("r"); if (it != json.end()) json.erase(it); json << reactions; dbei.flushJson(); db_event_edit(dbei.getEvent(), &dbei, true); } ///////////////////////////////////////////////////////////////////////////////////////// static char *sttBotIds[] = { "replies_bot_chat_id", "group_anonymous_bot_user_id", "channel_bot_user_id", "anti_spam_bot_user_id", }; void CTelegramProto::ProcessOption(TD::updateOption *pObj) { TD::int53 iValue = 0; if (pObj->value_->get_id() == TD::optionValueInteger::ID) iValue = ((TD::optionValueInteger *)pObj->value_.get())->value_; if (pObj->name_ == "my_id") { m_iOwnId = iValue; SetId(0, m_iOwnId); if (m_iSavedMessages != 0) return; if (auto *pUser = FindUser(iValue)) { m_iSavedMessages = pUser->hContact; pUser->hContact = 0; } else { m_iSavedMessages = db_add_contact(); Proto_AddToContact(m_iSavedMessages, m_szModuleName); SetId(m_iSavedMessages, m_iOwnId); Clist_SetGroup(m_iSavedMessages, m_wszDefaultGroup); pUser = new TG_USER(m_iOwnId, 0); m_arUsers.insert(pUser); m_arChats.insert(pUser); } setWString(m_iSavedMessages, "Nick", TranslateT("Saved messages")); return; } for (auto &it : sttBotIds) if (pObj->name_ == it) { m_botIds.push_back(iValue); return; } } ///////////////////////////////////////////////////////////////////////////////////////// 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: if (m_iTimeDiff1) { setWord(pUser->hContact, "Status", m_iStatus1); pUser->m_timer1 = time(0); } else setWord(pUser->hContact, "Status", ID_STATUS_OFFLINE); break; default: debugLogA("!!!!! Unknown status packet, report it to the developers"); } } } void CTelegramProto::ProcessUser(TD::updateUser *pObj) { auto *pUser = pObj->user_.get(); bool bIsMe = pUser->id_ == m_iOwnId; auto typeID = (pUser->type_) ? pUser->type_->get_id() : 0; if (!bIsMe && !pUser->is_contact_) { switch (typeID) { case TD::userTypeDeleted::ID: return; case TD::userTypeRegular::ID: auto *pu = AddFakeUser(pUser->id_, false); if (pu->hContact != INVALID_CONTACT_ID) RemoveFromClist(pu); 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; } for (auto &it : m_botIds) if (it == pUser->id_) { if (auto *pu = FindUser(it)) { Contact::Hide(pu->hContact); Contact::RemoveFromList(pu->hContact); } return; } std::string szFirstName = pUser->first_name_, szLastName = pUser->last_name_; if (szLastName.empty()) { size_t p = szFirstName.rfind(' '); if (p != -1) { szLastName = szFirstName.substr(p + 1); szFirstName = szFirstName.substr(0, p); } } // if that's just a bot, keep it as a virtual contact in memory if (typeID == TD::userTypeBot::ID) { auto *pu = AddFakeUser(pUser->id_, false); if (pu->hContact == INVALID_CONTACT_ID) { pu->wszFirstName = Utf2T(szFirstName.c_str()); pu->wszLastName = Utf2T(szLastName.c_str()); if (pUser->usernames_) pu->wszNick = Utf2T(pUser->usernames_->editable_username_.c_str()); return; } } auto *pu = AddUser(pUser->id_, false); if (szFirstName.empty()) delSetting(pu->hContact, "FirstName"); else setUString(pu->hContact, "FirstName", szFirstName.c_str()); if (szLastName.empty()) delSetting(pu->hContact, "LastName"); else setUString(pu->hContact, "LastName", szLastName.c_str()); if (pu->hContact) UpdateString(pu->hContact, "Phone", pUser->phone_number_); if (pUser->usernames_) UpdateString(pu->hContact, "Nick", pUser->usernames_->editable_username_); Contact::PutOnList(pu->hContact); if (bIsMe) pu->wszNick = ptrW(Contact::GetInfo(CNF_DISPLAY, 0, m_szModuleName)); if (pUser->is_premium_) ExtraIcon_SetIconByName(g_plugin.m_hIcon, pu->hContact, "tg_premium"); else if (typeID == TD::userTypeBot::ID) { pu->isBot = true; ExtraIcon_SetIconByName(g_plugin.m_hIcon, pu->hContact, "tg_bot"); } else ExtraIcon_SetIconByName(g_plugin.m_hIcon, pu->hContact, nullptr); if (auto *pPhoto = pUser->profile_photo_.get()) ProcessAvatar(pPhoto->small_.get(), pu); 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_); } } }