/* * The ManaPlus Client * Copyright (C) 2004-2009 The Mana World Development Team * Copyright (C) 2009-2010 The Mana Developers * Copyright (C) 2011-2015 The ManaPlus Developers * * This file is part of The ManaPlus Client. * * 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; either version 2 of the License, or * any later version. * * 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 "net/tmwa/loginhandler.h" #include "client.h" #include "logger.h" #include "net/serverfeatures.h" #include "net/tmwa/messageout.h" #include "net/tmwa/network.h" #include "net/tmwa/protocol.h" #include "utils/gettext.h" #include "debug.h" extern Net::LoginHandler *loginHandler; namespace TmwAthena { extern ServerInfo charServer; enum ServerFlags { FLAG_REGISTRATION = 1 }; LoginHandler::LoginHandler() : MessageHandler(), Ea::LoginHandler() { static const uint16_t _messages[] = { SMSG_UPDATE_HOST, SMSG_LOGIN_DATA, SMSG_LOGIN_ERROR, SMSG_CHAR_PASSWORD_RESPONSE, SMSG_SERVER_VERSION_RESPONSE, 0 }; handledMessages = _messages; loginHandler = this; } LoginHandler::~LoginHandler() { } void LoginHandler::handleMessage(Net::MessageIn &msg) { BLOCK_START("LoginHandler::handleMessage") switch (msg.getId()) { case SMSG_CHAR_PASSWORD_RESPONSE: processCharPasswordResponse(msg); break; case SMSG_UPDATE_HOST: processUpdateHost(msg); break; case SMSG_LOGIN_DATA: processLoginData(msg); break; case SMSG_LOGIN_ERROR: processLoginError(msg); break; case SMSG_SERVER_VERSION_RESPONSE: processServerVersion(msg); break; default: break; } BLOCK_END("LoginHandler::handleMessage") } void LoginHandler::connect() { if (!mNetwork) return; mNetwork->connect(mServer); if (serverFeatures->haveServerVersion()) createOutPacket(CMSG_SERVER_VERSION_REQUEST); } bool LoginHandler::isConnected() const { if (!mNetwork) return false; return mVersionResponse && mNetwork->isConnected(); } void LoginHandler::disconnect() { if (mNetwork && mNetwork->getServer() == mServer) mNetwork->disconnect(); } void LoginHandler::changePassword(const std::string &restrict oldPassword, const std::string &restrict newPassword) const { createOutPacket(CMSG_CHAR_PASSWORD_CHANGE); outMsg.writeStringNoLog(oldPassword, 24, "old password"); outMsg.writeStringNoLog(newPassword, 24, "new password"); } void LoginHandler::sendLoginRegister(const std::string &restrict username, const std::string &restrict password, const std::string &restrict email A_UNUSED) const { createOutPacket(CMSG_LOGIN_REGISTER); if (serverVersion > 0) { outMsg.writeInt32(CLIENT_PROTOCOL_VERSION, "client protocol version"); } else { outMsg.writeInt32(CLIENT_TMW_PROTOCOL_VERSION, "client protocol version"); } outMsg.writeString(username, 24, "login"); outMsg.writeStringNoLog(password, 24, "password"); /* * eAthena calls the last byte "client version 2", but it isn't used at * at all. We're retasking it, as a bit mask: * 0 - can handle the 0x63 "update host" packet * 1 - defaults to the first char-server (instead of the last) */ outMsg.writeInt8(0x03, "flags"); } ServerInfo *LoginHandler::getCharServer() const { return &charServer; } void LoginHandler::requestUpdateHosts() { } void LoginHandler::processServerVersion(Net::MessageIn &msg) { const uint8_t b1 = msg.readUInt8("b1"); // -1 const uint8_t b2 = msg.readUInt8("b2"); // E const uint8_t b3 = msg.readUInt8("b3"); // V const uint8_t b4 = msg.readUInt8("b4"); // L if (b1 == 255) { // old TMWA const unsigned int options = msg.readInt32("options"); mRegistrationEnabled = options & FLAG_REGISTRATION; serverVersion = 0; tmwServerVersion = 0; } else if (b1 >= 0x0d) { // new TMWA const unsigned int options = msg.readInt32("options"); mRegistrationEnabled = options & FLAG_REGISTRATION; serverVersion = 0; tmwServerVersion = (b1 << 16) | (b2 << 8) | b3; } else { // eAthena const unsigned int options = msg.readInt32("options"); mRegistrationEnabled = options & FLAG_REGISTRATION; serverVersion = 0; tmwServerVersion = 0; } if (serverVersion > 0) logger->log("Evol server version: %d", serverVersion); else if (tmwServerVersion > 0) logger->log("Tmw server version: x%06x", tmwServerVersion); else logger->log("Server without version"); if (serverVersion < 5) { if (client->getState() != STATE_LOGIN) client->setState(STATE_LOGIN); } // Leave this last mVersionResponse = true; } int LoginHandler::supportedOptionalActions() const { return Net::RegistrationOptions::SetGenderOnRegister; } void LoginHandler::sendVersion() const { } void LoginHandler::processCharPasswordResponse(Net::MessageIn &msg) { // 0: acc not found, 1: success, 2: password mismatch, 3: pass too short const uint8_t errMsg = msg.readUInt8("result code"); // Successful pass change if (errMsg == 1) { client->setState(STATE_CHANGEPASSWORD_SUCCESS); } // pass change failed else { switch (errMsg) { case 0: errorMessage = // TRANSLATORS: error message _("Account was not found. Please re-login."); break; case 2: // TRANSLATORS: error message errorMessage = _("Old password incorrect."); break; case 3: // TRANSLATORS: error message errorMessage = _("New password too short."); break; default: // TRANSLATORS: error message errorMessage = _("Unknown error."); break; } client->setState(STATE_ACCOUNTCHANGE_ERROR); } } void LoginHandler::ping() const { } } // namespace TmwAthena