/* Copyright (C) 2012-22 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 __cdecl CMTProto::ServerThread(void *) { m_bRunning = true; m_bTerminated = m_bAuthorized = false; SendQuery(new td::td_api::getOption("version")); while (!m_bTerminated) { ProcessResponse(m_pClientMmanager->receive(10)); } m_bRunning = false; } void CMTProto::LogOut() { if (m_bTerminated) return; debugLogA("CMTProto::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; setAllContactStatuses(ID_STATUS_OFFLINE, false); } void CMTProto::OnLoggedIn() { m_bAuthorized = true; debugLogA("CMTProto::OnLoggedIn"); ProtoBroadcastAck(0, ACKTYPE_STATUS, ACKRESULT_SUCCESS, (HANDLE)m_iStatus, m_iDesiredStatus); m_iStatus = m_iDesiredStatus; } /////////////////////////////////////////////////////////////////////////////// INT_PTR CALLBACK CMTProto::EnterPhoneCode(void *param) { auto *ppro = (CMTProto *)param; ENTER_STRING es = {}; es.szModuleName = ppro->m_szModuleName; es.caption = TranslateT("Enter secret code sent to your phone"); if (EnterString(&es)) { ppro->SendQuery(new td::td_api::checkAuthenticationCode(_T2A(es.ptszResult).get()), &CMTProto::OnUpdateAuth); mir_free(es.ptszResult); } else ppro->LogOut(); return 0; } INT_PTR CALLBACK CMTProto::EnterPassword(void *param) { auto *ppro = (CMTProto *)param; CMStringW wszTitle(TranslateT("Enter password")); auto *pAuth = (td::td_api::authorizationStateWaitPassword *)ppro->pAuthState.get(); if (!pAuth->password_hint_.empty()) wszTitle.AppendFormat(TranslateT(" (hint: %s)"), Utf2T(pAuth->password_hint_.c_str()).get()); ENTER_STRING es = {}; es.szModuleName = ppro->m_szModuleName; es.caption = wszTitle; es.type = ESF_PASSWORD; if (EnterString(&es)) { ppro->SendQuery(new td::td_api::checkAuthenticationPassword(_T2A(es.ptszResult).get()), &CMTProto::OnUpdateAuth); mir_free(es.ptszResult); } else ppro->LogOut(); return 0; } void CMTProto::ProcessAuth(td::td_api::updateAuthorizationState *pObj) { pAuthState = std::move(pObj->authorization_state_); switch (pAuthState->get_id()) { case td::td_api::authorizationStateWaitTdlibParameters::ID: { char text[100]; Miranda_GetVersionText(text, sizeof(text)); CMStringW wszPath(GetProtoFolder()); auto *request = new td::td_api::setTdlibParameters(); request->database_directory_ = T2Utf(wszPath).get(); request->use_message_database_ = false; request->use_secret_chats_ = true; request->api_id_ = 94575; request->api_hash_ = "a3406de8d171bb422bb6ddf3bbd800e2"; request->system_language_code_ = "en"; request->device_model_ = "Miranda NG"; request->application_version_ = text; request->enable_storage_optimizer_ = true; SendQuery(request, &CMTProto::OnUpdateAuth); } break; case td::td_api::authorizationStateWaitPhoneNumber::ID: SendQuery(new td::td_api::setAuthenticationPhoneNumber(_T2A(m_szOwnPhone).get(), nullptr), &CMTProto::OnUpdateAuth); break; case td::td_api::authorizationStateWaitCode::ID: CallFunctionSync(EnterPhoneCode, this); break; case td::td_api::authorizationStateWaitPassword::ID: CallFunctionSync(EnterPassword, this); break; case td::td_api::authorizationStateReady::ID: OnLoggedIn(); break; case td::td_api::authorizationStateLoggingOut::ID: debugLogA("Server required us to log out, exiting"); LogOut(); break; case td::td_api::authorizationStateClosing::ID: debugLogA("Connection terminated, exiting"); LogOut(); break; } } void CMTProto::OnUpdateAuth(td::ClientManager::Response &response) { if (response.object->get_id() == td::td_api::error::ID) { auto *pError = (td::td_api::error*)response.object.get(); debugLogA("error happened: %s", to_string(*pError).c_str()); if (pError->message_ == "PHONE_CODE_EXPIRED") Popup(0, TranslateT("Phone code expired"), TranslateT("Error")); else if(pError->message_ == "INVALID_PHONE_CODE") Popup(0, TranslateT("Invalid phone code"), TranslateT("Error")); pAuthState = std::move(nullptr); LogOut(); } } /////////////////////////////////////////////////////////////////////////////// void CMTProto::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) { auto *p = m_arRequests.find((TG_REQUEST *)&response.request_id); if (p) { (this->*p->pHandler)(response); m_arRequests.remove(p); } return; } switch (response.object->get_id()) { case td::td_api::updateAuthorizationState::ID: ProcessAuth((td::td_api::updateAuthorizationState *)response.object.get()); break; } } void CMTProto::SendQuery(td::td_api::Function *pFunc, TG_QUERY_HANDLER pHandler) { int queryId = ++m_iQueryId; m_pClientMmanager->send(m_iClientId, queryId, td::td_api::object_ptr(pFunc)); if (pHandler) m_arRequests.insert(new TG_REQUEST(queryId, pHandler)); }