mirror of
https://github.com/anope/anope.git
synced 2026-06-12 18:54:47 +02:00
6e0f0b8896
Using HMAC instead of changing the IV is a lot safer.
191 lines
5.4 KiB
C++
191 lines
5.4 KiB
C++
/* Module for providing bcrypt hashing
|
|
*
|
|
* (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 <climits>
|
|
#include <random>
|
|
|
|
#include "sha2/sha2.c"
|
|
|
|
#include "module.h"
|
|
#include "modules/encryption.h"
|
|
|
|
template <typename SHAContext,
|
|
void (* SHAInit)(SHAContext *),
|
|
void (* SHAUpdate)(SHAContext *, const unsigned char *, unsigned int),
|
|
void (* SHAFinal)(SHAContext *, unsigned char *)>
|
|
class SHA2Context final
|
|
: public Encryption::Context
|
|
{
|
|
private:
|
|
SHAContext context;
|
|
const size_t digest_size;
|
|
|
|
public:
|
|
SHA2Context(size_t ds)
|
|
: digest_size(ds)
|
|
{
|
|
SHAInit(&context);
|
|
}
|
|
|
|
void Update(const unsigned char *data, size_t len) override
|
|
{
|
|
SHAUpdate(&context, data, len);
|
|
}
|
|
|
|
Anope::string Finalize() override
|
|
{
|
|
std::vector<unsigned char> digest(digest_size);
|
|
SHAFinal(&context, digest.data());
|
|
return Anope::string(reinterpret_cast<const char *>(digest.data()), digest.size());
|
|
}
|
|
};
|
|
|
|
template <typename SHAContext,
|
|
void (* SHAInit)(SHAContext *),
|
|
void (* SHAUpdate)(SHAContext *, const unsigned char *, unsigned int),
|
|
void (* SHAFinal)(SHAContext *, unsigned char *)>
|
|
class SHA2Provider final
|
|
: public Encryption::Provider
|
|
{
|
|
public:
|
|
SHA2Provider(Module *creator, const Anope::string &algorithm, size_t bs, size_t ds)
|
|
: Encryption::Provider(creator, algorithm, bs, ds)
|
|
{
|
|
}
|
|
|
|
std::unique_ptr<Encryption::Context> CreateContext() override
|
|
{
|
|
return std::make_unique<SHA2Context<SHAContext, SHAInit, SHAUpdate, SHAFinal>>(this->digest_size);
|
|
}
|
|
};
|
|
|
|
class ESHA2 final
|
|
: public Module
|
|
{
|
|
private:
|
|
Encryption::Provider *defaultprovider = nullptr;
|
|
SHA2Provider<sha224_ctx, sha224_init, sha224_update, sha224_final> sha224provider;
|
|
SHA2Provider<sha256_ctx, sha256_init, sha256_update, sha256_final> sha256provider;
|
|
SHA2Provider<sha384_ctx, sha384_init, sha384_update, sha384_final> sha384provider;
|
|
SHA2Provider<sha512_ctx, sha512_init, sha512_update, sha512_final> sha512provider;
|
|
|
|
Anope::string GenerateKey(size_t keylen)
|
|
{
|
|
static std::random_device device;
|
|
static std::mt19937 engine(device());
|
|
static std::uniform_int_distribution<int> dist(CHAR_MIN, CHAR_MAX);
|
|
Anope::string keybuf(keylen, ' ');
|
|
for (size_t i = 0; i < keylen; ++i)
|
|
keybuf[i] = static_cast<char>(dist(engine));
|
|
return keybuf;
|
|
}
|
|
|
|
Encryption::Provider *GetAlgorithm(const Anope::string &algorithm)
|
|
{
|
|
if (algorithm == "sha224")
|
|
return &sha224provider;
|
|
if (algorithm == "sha256")
|
|
return &sha256provider;
|
|
if (algorithm == "sha384")
|
|
return &sha384provider;
|
|
if (algorithm == "sha512")
|
|
return &sha512provider;
|
|
return nullptr;
|
|
}
|
|
|
|
Anope::string HMAC(Encryption::Provider *provider, const Anope::string &key, const Anope::string &data)
|
|
{
|
|
auto keybuf = key.length() > provider->block_size ? provider->Encrypt(key) : key;
|
|
keybuf.resize(provider->block_size);
|
|
|
|
Anope::string hmac1;
|
|
Anope::string hmac2;
|
|
for (size_t i = 0; i < provider->block_size; ++i)
|
|
{
|
|
hmac1.push_back(static_cast<char>(keybuf[i] ^ 0x5C));
|
|
hmac2.push_back(static_cast<char>(keybuf[i] ^ 0x36));
|
|
}
|
|
hmac2.append(data);
|
|
hmac1.append(provider->Encrypt(hmac2));
|
|
|
|
return provider->Encrypt(hmac1);
|
|
}
|
|
|
|
public:
|
|
ESHA2(const Anope::string &modname, const Anope::string &creator)
|
|
: Module(modname, creator, ENCRYPTION | VENDOR)
|
|
, sha224provider(this, "sha224", SHA224_BLOCK_SIZE, SHA224_DIGEST_SIZE)
|
|
, sha256provider(this, "sha256", SHA256_BLOCK_SIZE, SHA256_DIGEST_SIZE)
|
|
, sha384provider(this, "sha384", SHA384_BLOCK_SIZE, SHA384_DIGEST_SIZE)
|
|
, sha512provider(this, "sha512", SHA512_BLOCK_SIZE, SHA512_DIGEST_SIZE)
|
|
{
|
|
}
|
|
|
|
void OnReload(Configuration::Conf *conf) override
|
|
{
|
|
this->defaultprovider = GetAlgorithm(Config->GetModule(this)->Get<const Anope::string>("algorithm", "sha256"));
|
|
}
|
|
|
|
EventReturn OnEncrypt(const Anope::string &src, Anope::string &dest) override
|
|
{
|
|
if (!defaultprovider)
|
|
return EVENT_CONTINUE;
|
|
|
|
auto key = GenerateKey(defaultprovider->digest_size);
|
|
auto hmac = HMAC(defaultprovider, key, src);
|
|
auto enc = "hmac-" + defaultprovider->name + ":" + Anope::Hex(hmac) + ":" + Anope::Hex(key);
|
|
Log(LOG_DEBUG_2) << "(enc_sha2) hashed password from [" << src << "] to [" << enc << "]";
|
|
dest = enc;
|
|
return EVENT_ALLOW;
|
|
}
|
|
|
|
void OnCheckAuthentication(User *, IdentifyRequest *req) override
|
|
{
|
|
const auto *na = NickAlias::Find(req->GetAccount());
|
|
if (!na)
|
|
return;
|
|
|
|
NickCore *nc = na->nc;
|
|
auto apos = nc->pass.find(':');
|
|
if (apos == Anope::string::npos)
|
|
return;
|
|
|
|
Anope::string hash_method(nc->pass.begin(), nc->pass.begin() + apos);
|
|
if (hash_method.compare(0, 5, "hmac-", 5))
|
|
return; // Not a HMAC hash.
|
|
|
|
auto provider = GetAlgorithm(hash_method.substr(5));
|
|
if (!provider)
|
|
return; // Not a hash for this module.
|
|
|
|
auto bpos = nc->pass.find(':', apos + 1);
|
|
if (bpos == Anope::string::npos)
|
|
return; // No HMAC key.
|
|
|
|
Anope::string pass_hex(nc->pass.begin() + apos + 1, nc->pass.begin() + bpos);
|
|
Anope::string key_hex(nc->pass.begin() + bpos + 1, nc->pass.end());
|
|
Anope::string key;
|
|
Anope::Unhex(key_hex, key);
|
|
|
|
auto enc = Anope::Hex(HMAC(provider, key, req->GetPassword()));
|
|
if (pass_hex.equals_cs(enc))
|
|
{
|
|
// If we are NOT the first encryption module or the algorithm is
|
|
// different we want to re-encrypt the password with the primary
|
|
// encryption method.
|
|
if (ModuleManager::FindFirstOf(ENCRYPTION) != this || provider != defaultprovider)
|
|
Anope::Encrypt(req->GetPassword(), nc->pass);
|
|
req->Success(this);
|
|
}
|
|
}
|
|
};
|
|
|
|
MODULE_INIT(ESHA2)
|