/* This module generates and compares password hashes using SHA256 algorithms. * * (C) 2003-2025 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]); } Anope::string EncryptInternal(const Anope::string &src) { 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)); return "sha256:" + Anope::Hex(hash) + ":" + GetIVString(); } 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"); } void OnCheckAuthentication(User *, IdentifyRequest *req) override { const auto *na = NickAlias::Find(req->GetAccount()); if (!na) return; NickCore *nc = na->nc; auto 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; auto enc = EncryptInternal(req->GetPassword()); if (nc->pass.equals_cs(enc)) { // If we are NOT the first encryption module we want to re-encrypt // the password with the primary encryption method. if (ModuleManager::FindFirstOf(ENCRYPTION) != this) Anope::Encrypt(req->GetPassword(), nc->pass); req->Success(this); } } }; MODULE_INIT(ESHA256)