/* This module generates and compares password hashes using SHA256 algorithms. * * (C) 2003-2024 Anope Team * Contact us at team@anope.org * * This program is free but copyrighted software; see the file COPYING for * details. * */ #include "sha2/sha2.c" #include "module.h" class ESHA256 final : public Module { private: unsigned iv[8]; bool use_iv; /* initializes the IV with a new random value */ void NewRandomIV() { for (auto &ivsegment : iv) ivsegment = static_cast(Anope::RandomNumber()); } /* returns the IV as base64-encrypted string */ Anope::string GetIVString() { char buf[33]; for (int i = 0; i < 8; ++i) UNPACK32(iv[i], reinterpret_cast(&buf[i << 2])); buf[32] = '\0'; return Anope::Hex(buf, 32); } /* splits the appended IV from the password string so it can be used for the next encryption */ /* password format: :: */ void GetIVFromPass(const Anope::string &password) { size_t pos = password.find(':'); Anope::string buf = password.substr(password.find(':', pos + 1) + 1, password.length()); char buf2[33]; Anope::Unhex(buf, buf2, sizeof(buf2)); for (int i = 0 ; i < 8; ++i) PACK32(reinterpret_cast(&buf2[i << 2]), &iv[i]); } public: ESHA256(const Anope::string &modname, const Anope::string &creator) : Module(modname, creator, ENCRYPTION | VENDOR) { use_iv = false; if (ModuleManager::FindFirstOf(ENCRYPTION) == this) throw ModuleException("enc_sha256 is deprecated and can not be used as a primary encryption method"); } EventReturn OnEncrypt(const Anope::string &src, Anope::string &dest) override { if (!use_iv) NewRandomIV(); else use_iv = false; sha256_ctx ctx; sha256_init(&ctx); for (size_t i = 0; i < 8; ++i) ctx.h[i] = iv[i]; sha256_update(&ctx, reinterpret_cast(src.data()), src.length()); unsigned char digest[SHA256_DIGEST_SIZE]; sha256_final(&ctx, digest); Anope::string hash(reinterpret_cast(&digest), sizeof(digest)); std::stringstream buf; buf << "sha256:" << Anope::Hex(hash) << ":" << GetIVString(); Log(LOG_DEBUG_2) << "(enc_sha256) hashed password from [" << src << "] to [" << buf.str() << " ]"; dest = buf.str(); return EVENT_ALLOW; } void OnCheckAuthentication(User *, IdentifyRequest *req) override { const NickAlias *na = NickAlias::Find(req->GetAccount()); if (na == NULL) return; NickCore *nc = na->nc; size_t pos = nc->pass.find(':'); if (pos == Anope::string::npos) return; Anope::string hash_method(nc->pass.begin(), nc->pass.begin() + pos); if (!hash_method.equals_cs("sha256")) return; GetIVFromPass(nc->pass); use_iv = true; Anope::string buf; this->OnEncrypt(req->GetPassword(), buf); if (nc->pass.equals_cs(buf)) { /* if we are NOT the first module in the list or we are using a default IV * we want to re-encrypt the pass with the new encryption */ if (ModuleManager::FindFirstOf(ENCRYPTION) != this || !memcmp(iv, sha256_h0, 8)) Anope::Encrypt(req->GetPassword(), nc->pass); req->Success(this); } } }; MODULE_INIT(ESHA256)