From a9ee4e0cb65be5b61fa785f191edf008d91e63b1 Mon Sep 17 00:00:00 2001 From: insunaa Date: Sun, 20 Oct 2024 17:46:09 +0200 Subject: [PATCH] Auth: Enable 2FA on Classic Co-authored-by: Chaosvex --- src/realmd/AuthSocket.cpp | 110 +++++++++++++++++++++++++++++++++++--- src/realmd/AuthSocket.h | 6 +++ 2 files changed, 109 insertions(+), 7 deletions(-) diff --git a/src/realmd/AuthSocket.cpp b/src/realmd/AuthSocket.cpp index e24e22b02ed..49bf19a543f 100644 --- a/src/realmd/AuthSocket.cpp +++ b/src/realmd/AuthSocket.cpp @@ -87,11 +87,11 @@ struct sAuthLogonChallengeHeader uint16 size; }; -typedef struct AUTH_LOGON_PIN_DATA_C +struct sAuthLogonPinData_C { uint8 salt[16]; uint8 hash[20]; -} sAuthLogonPinData_C; +}; // typedef sAuthLogonChallenge_C sAuthReconnectChallenge_C; /* @@ -117,6 +117,11 @@ struct sAuthLogonProof_C uint8 crc_hash[20]; uint8 number_of_keys; uint8 securityFlags; // 0x00-0x04 + sAuthLogonPinData_C pinData; // PINData for PIN authentication (classic only). Conditional read if PIN was requested + static size_t getSize(bool withPin) + { + return sizeof(sAuthLogonProof_C) - (withPin ? 0 : sizeof(sAuthLogonPinData_C)); + } }; /* typedef struct @@ -492,13 +497,20 @@ bool AuthSocket::_HandleLogonChallenge() if (!self->_token.empty() && self->_build >= 8606) // authenticator was added in 2.4.3 securityFlags = SECURITY_FLAG_AUTHENTICATOR; + if (!self->_token.empty() && self->_build <= 6141) + securityFlags = SECURITY_FLAG_PIN; + *pkt << uint8(securityFlags); // security flags (0x0...0x04) if (securityFlags & SECURITY_FLAG_PIN) // PIN input { - *pkt << uint32(0); - *pkt << uint64(0); - *pkt << uint64(0); + uint32 gridSeedPkt = self->m_gridSeed = static_cast(0); + EndianConvert(gridSeedPkt); + self->m_serverSecuritySalt.SetRand(16 * 8); // 16 bytes random + self->m_promptPin = true; + + *pkt << gridSeedPkt; + pkt->append(self->m_serverSecuritySalt.AsByteArray(16).data(), 16); } if (securityFlags & SECURITY_FLAG_UNK) // Matrix input @@ -539,7 +551,8 @@ bool AuthSocket::_HandleLogonProof() DEBUG_LOG("Entering _HandleLogonProof"); ///- Read the packet std::shared_ptr lp = std::make_shared(); - Read((char*)lp.get(), sizeof(sAuthLogonProof_C), [self = shared_from_this(), lp](const boost::system::error_code& error, std::size_t read) + auto size = sAuthLogonProof_C::getSize(m_promptPin); + Read((char*)lp.get(), size, [self = shared_from_this(), lp](const boost::system::error_code& error, std::size_t read) { if (error) { @@ -578,7 +591,7 @@ bool AuthSocket::_HandleLogonProof() ///- Check if SRP6 results match (password is correct), else send an error if (!self->srp.Proof(lp->M1, 20)) { - if (lp->securityFlags & SECURITY_FLAG_AUTHENTICATOR || !self->_token.empty()) + if (self->_build > 6141 && (lp->securityFlags & SECURITY_FLAG_AUTHENTICATOR || !self->_token.empty())) { std::shared_ptr pinCount = std::make_shared(); self->Read((char*)pinCount.get(), sizeof(uint8), [self, pinCount, lp](const boost::system::error_code& error, std::size_t read) @@ -614,6 +627,17 @@ bool AuthSocket::_HandleLogonProof() return; } + if ((lp->securityFlags & SECURITY_FLAG_PIN) && !self->_token.empty()) + { + int32 serverToken = self->generateToken(self->_token.c_str()); + if (!self->VerifyPinData(serverToken, lp->pinData)) + { + BASIC_LOG("[AuthChallenge] Account %s tried to login with wrong pincode!", self->_login.c_str()); + self->Write(logonProofUnknownAccount, sizeof(logonProofUnknownAccount), [self](const boost::system::error_code& error, std::size_t read) {}); + return; + } + } + self->verifyVersionAndFinalizeAuthentication(lp); } else @@ -1040,6 +1064,78 @@ bool AuthSocket::_HandleXferAccept() return true; } +// Verify PIN entry data +bool AuthSocket::VerifyPinData(uint32 pin, const sAuthLogonPinData_C& clientData) +{ + // remap the grid to match the client's layout + std::vector grid { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9 }; + std::vector remappedGrid(grid.size()); + + uint8* remappedIndex = remappedGrid.data(); + uint32 seed = m_gridSeed; + + for (size_t i = grid.size(); i > 0; --i) + { + auto remainder = seed % i; + seed /= i; + *remappedIndex = grid[remainder]; + + size_t copySize = i; + copySize -= remainder; + --copySize; + + uint8* srcPtr = grid.data() + remainder + 1; + uint8* dstPtr = grid.data() + remainder; + + std::copy(srcPtr, srcPtr + copySize, dstPtr); + ++remappedIndex; + } + + // convert the PIN to bytes (for ex. '1234' to {1, 2, 3, 4}) + std::vector pinBytes; + + while (pin != 0) + { + pinBytes.push_back(pin % 10); + pin /= 10; + } + + std::reverse(pinBytes.begin(), pinBytes.end()); + + // validate PIN length + if (pinBytes.size() < 4 || pinBytes.size() > 10) + return false; // PIN outside of expected range + + // remap the PIN to calculate the expected client input sequence + for (size_t i = 0; i < pinBytes.size(); ++i) + { + auto index = std::find(remappedGrid.begin(), remappedGrid.end(), pinBytes[i]); + pinBytes[i] = std::distance(remappedGrid.begin(), index); + } + + // convert PIN bytes to their ASCII values + for (size_t i = 0; i < pinBytes.size(); ++i) + pinBytes[i] += 0x30; + + // validate the PIN, x = H(client_salt | H(server_salt | ascii(pin_bytes))) + Sha1Hash sha; + sha.UpdateData(m_serverSecuritySalt.AsByteArray()); + sha.UpdateData(pinBytes.data(), pinBytes.size()); + sha.Finalize(); + + BigNumber hash, clientHash; + hash.SetBinary(sha.GetDigest(), sha.GetLength()); + clientHash.SetBinary(clientData.hash, 20); + + sha.Initialize(); + sha.UpdateData(clientData.salt, sizeof(clientData.salt)); + sha.UpdateData(hash.AsByteArray()); + sha.Finalize(); + hash.SetBinary(sha.GetDigest(), sha.GetLength()); + + return !memcmp(hash.AsDecStr(), clientHash.AsDecStr(), 20); +} + void AuthSocket::verifyVersionAndFinalizeAuthentication(std::shared_ptr lp) { if (!VerifyVersion(lp->A, sizeof(lp->A), lp->crc_hash, false)) diff --git a/src/realmd/AuthSocket.h b/src/realmd/AuthSocket.h index 9a851e6bdbc..54bc5d81738 100644 --- a/src/realmd/AuthSocket.h +++ b/src/realmd/AuthSocket.h @@ -38,6 +38,7 @@ #define HMAC_RES_SIZE 20 struct sAuthLogonProof_C; +struct sAuthLogonPinData_C; class AuthSocket : public MaNGOS::AsyncSocket { @@ -50,6 +51,7 @@ class AuthSocket : public MaNGOS::AsyncSocket void SendProof(Sha1Hash sha); void LoadRealmlist(ByteBuffer& pkt, uint32 acctid, uint8 accountSecurityLevel = 0); + bool VerifyPinData(uint32 pin, const sAuthLogonPinData_C& clientData); int32 generateToken(char const* b32key); uint8 getEligibleRealmCount(uint8 accountSecurityLevel); @@ -94,6 +96,10 @@ class AuthSocket : public MaNGOS::AsyncSocket uint16 _build; AccountTypes _accountSecurityLevel; + BigNumber m_serverSecuritySalt; + uint32 m_gridSeed = 0; + bool m_promptPin = false; + boost::asio::deadline_timer m_timeoutTimer; virtual bool ProcessIncomingData() override;