mirror of
https://github.com/anope/anope.git
synced 2026-06-25 00:26:37 +02:00
334 lines
8.6 KiB
C++
334 lines
8.6 KiB
C++
#include "module.h"
|
|
#include "nickserv.h"
|
|
#include "ldap.h"
|
|
|
|
static Module *me;
|
|
|
|
static Anope::string basedn;
|
|
static Anope::string search_filter;
|
|
static Anope::string object_class;
|
|
static Anope::string email_attribute;
|
|
static Anope::string username_attribute;
|
|
|
|
struct IdentifyInfo
|
|
{
|
|
dynamic_reference<User> user;
|
|
IdentifyRequest *req;
|
|
service_reference<LDAPProvider> lprov;
|
|
bool admin_bind;
|
|
Anope::string dn;
|
|
|
|
IdentifyInfo(User *u, IdentifyRequest *r, service_reference<LDAPProvider> &lp) : user(u), req(r), lprov(lp), admin_bind(true)
|
|
{
|
|
req->Hold(me);
|
|
}
|
|
|
|
~IdentifyInfo()
|
|
{
|
|
req->Release(me);
|
|
}
|
|
};
|
|
|
|
class IdentifyInterface : public LDAPInterface
|
|
{
|
|
std::map<LDAPQuery, IdentifyInfo *> requests;
|
|
|
|
public:
|
|
IdentifyInterface(Module *m) : LDAPInterface(m) { }
|
|
|
|
void Add(LDAPQuery id, IdentifyInfo *ii)
|
|
{
|
|
std::map<LDAPQuery, IdentifyInfo *>::iterator it = this->requests.find(id);
|
|
if (it != this->requests.end())
|
|
delete it->second;
|
|
this->requests[id] = ii;
|
|
}
|
|
|
|
void OnResult(const LDAPResult &r) anope_override
|
|
{
|
|
std::map<LDAPQuery, IdentifyInfo *>::iterator it = this->requests.find(r.id);
|
|
if (it == this->requests.end())
|
|
return;
|
|
IdentifyInfo *ii = it->second;
|
|
this->requests.erase(it);
|
|
|
|
if (!ii->lprov)
|
|
{
|
|
delete ii;
|
|
return;
|
|
}
|
|
|
|
switch (r.type)
|
|
{
|
|
case LDAPResult::QUERY_SEARCH:
|
|
{
|
|
if (!r.empty())
|
|
{
|
|
try
|
|
{
|
|
const LDAPAttributes &attr = r.get(0);
|
|
ii->dn = attr.get("dn");
|
|
Log(LOG_DEBUG) << "m_ldap_authenticationn: binding as " << ii->dn;
|
|
LDAPQuery id = ii->lprov->Bind(this, ii->dn, ii->req->GetPassword());
|
|
this->Add(id, ii);
|
|
return;
|
|
}
|
|
catch (const LDAPException &ex)
|
|
{
|
|
Log(this->owner) << "m_ldap_authentication: Error binding after search: " << ex.GetReason();
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
case LDAPResult::QUERY_BIND:
|
|
{
|
|
if (ii->admin_bind)
|
|
{
|
|
Anope::string sf = search_filter.replace_all_cs("%account", ii->req->GetAccount()).replace_all_cs("%object_class", object_class);
|
|
try
|
|
{
|
|
Log(LOG_DEBUG) << "m_ldap_authentication: searching for " << sf;
|
|
LDAPQuery id = ii->lprov->Search(this, basedn, sf);
|
|
this->Add(id, ii);
|
|
ii->admin_bind = false;
|
|
return;
|
|
}
|
|
catch (const LDAPException &ex)
|
|
{
|
|
Log(this->owner) << "m_ldap_authentication: Unable to search for " << sf << ": " << ex.GetReason();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
NickAlias *na = findnick(ii->req->GetAccount());
|
|
if (na == NULL)
|
|
{
|
|
na = new NickAlias(ii->req->GetAccount(), new NickCore(ii->req->GetAccount()));
|
|
if (ii->user)
|
|
{
|
|
if (Config->NSAddAccessOnReg)
|
|
na->nc->AddAccess(create_mask(ii->user));
|
|
|
|
const BotInfo *bi = findbot(Config->NickServ);
|
|
if (bi)
|
|
ii->user->SendMessage(bi, _("Your account \002%s\002 has been successfully created."), na->nick.c_str());
|
|
}
|
|
}
|
|
na->nc->Extend("m_ldap_authentication_dn", new ExtensibleItemClass<Anope::string>(ii->dn));
|
|
|
|
ii->req->Success(me);
|
|
}
|
|
break;
|
|
}
|
|
default:
|
|
break;
|
|
}
|
|
|
|
delete ii;
|
|
}
|
|
|
|
void OnError(const LDAPResult &r) anope_override
|
|
{
|
|
std::map<LDAPQuery, IdentifyInfo *>::iterator it = this->requests.find(r.id);
|
|
if (it == this->requests.end())
|
|
return;
|
|
IdentifyInfo *ii = it->second;
|
|
this->requests.erase(it);
|
|
delete ii;
|
|
}
|
|
};
|
|
|
|
class OnIdentifyInterface : public LDAPInterface
|
|
{
|
|
std::map<LDAPQuery, Anope::string> requests;
|
|
|
|
public:
|
|
OnIdentifyInterface(Module *m) : LDAPInterface(m) { }
|
|
|
|
void Add(LDAPQuery id, const Anope::string &nick)
|
|
{
|
|
this->requests[id] = nick;
|
|
}
|
|
|
|
void OnResult(const LDAPResult &r) anope_override
|
|
{
|
|
std::map<LDAPQuery, Anope::string>::iterator it = this->requests.find(r.id);
|
|
if (it == this->requests.end())
|
|
return;
|
|
User *u = finduser(it->second);
|
|
this->requests.erase(it);
|
|
|
|
if (!u || !u->Account() || r.empty())
|
|
return;
|
|
|
|
try
|
|
{
|
|
const LDAPAttributes &attr = r.get(0);
|
|
Anope::string email = attr.get(email_attribute);
|
|
|
|
if (!email.equals_ci(u->Account()->email))
|
|
{
|
|
u->Account()->email = email;
|
|
BotInfo *bi = findbot(Config->NickServ);
|
|
if (bi)
|
|
u->SendMessage(bi, _("Your email has been updated to \002%s\002"), email.c_str());
|
|
Log(this->owner) << "m_ldap_authentication: Updated email address for " << u->nick << " (" << u->Account()->display << ") to " << email;
|
|
}
|
|
}
|
|
catch (const LDAPException &ex)
|
|
{
|
|
Log(this->owner) << "m_ldap_authentication: " << ex.GetReason();
|
|
}
|
|
}
|
|
|
|
void OnError(const LDAPResult &r) anope_override
|
|
{
|
|
this->requests.erase(r.id);
|
|
Log(this->owner) << "m_ldap_authentication: " << r.error;
|
|
}
|
|
};
|
|
|
|
class OnRegisterInterface : public LDAPInterface
|
|
{
|
|
public:
|
|
OnRegisterInterface(Module *m) : LDAPInterface(m) { }
|
|
|
|
void OnResult(const LDAPResult &r) anope_override
|
|
{
|
|
Log(this->owner) << "m_ldap_authentication: Successfully added newly created account to LDAP";
|
|
}
|
|
|
|
void OnError(const LDAPResult &r) anope_override
|
|
{
|
|
Log(this->owner) << "m_ldap_authentication: Error adding newly created account to LDAP: " << r.getError();
|
|
}
|
|
};
|
|
|
|
class NSIdentifyLDAP : public Module
|
|
{
|
|
service_reference<LDAPProvider> ldap;
|
|
IdentifyInterface iinterface;
|
|
OnIdentifyInterface oninterface;
|
|
OnRegisterInterface orinterface;
|
|
|
|
Anope::string password_attribute;
|
|
bool disable_register;
|
|
Anope::string disable_reason;
|
|
public:
|
|
NSIdentifyLDAP(const Anope::string &modname, const Anope::string &creator) :
|
|
Module(modname, creator, SUPPORTED), ldap("LDAPProvider", "ldap/main"), iinterface(this), oninterface(this), orinterface(this)
|
|
{
|
|
this->SetAuthor("Anope");
|
|
|
|
me = this;
|
|
|
|
Implementation i[] = { I_OnReload, I_OnPreCommand, I_OnCheckAuthentication, I_OnNickIdentify, I_OnNickRegister };
|
|
ModuleManager::Attach(i, this, sizeof(i) / sizeof(Implementation));
|
|
ModuleManager::SetPriority(this, PRIORITY_FIRST);
|
|
|
|
OnReload();
|
|
}
|
|
|
|
void OnReload() anope_override
|
|
{
|
|
ConfigReader config;
|
|
|
|
basedn = config.ReadValue("m_ldap_authentication", "basedn", "", 0);
|
|
search_filter = config.ReadValue("m_ldap_authentication", "search_filter", "", 0);
|
|
object_class = config.ReadValue("m_ldap_authentication", "object_class", "", 0);
|
|
username_attribute = config.ReadValue("m_ldap_authentication", "username_attribute", "", 0);
|
|
this->password_attribute = config.ReadValue("m_ldap_authentication", "password_attribute", "", 0);
|
|
email_attribute = config.ReadValue("m_ldap_authentication", "email_attribute", "", 0);
|
|
this->disable_register = config.ReadFlag("m_ldap_authentication", "disable_ns_register", "false", 0);
|
|
this->disable_reason = config.ReadValue("m_ldap_authentication", "disable_reason", "", 0);
|
|
}
|
|
|
|
EventReturn OnPreCommand(CommandSource &source, Command *command, std::vector<Anope::string> ¶ms) anope_override
|
|
{
|
|
if (this->disable_register && !this->disable_reason.empty() && command->name == "nickserv/register")
|
|
{
|
|
source.Reply(_(this->disable_reason.c_str()));
|
|
return EVENT_STOP;
|
|
}
|
|
|
|
return EVENT_CONTINUE;
|
|
}
|
|
|
|
void OnCheckAuthentication(User *u, IdentifyRequest *req) anope_override
|
|
{
|
|
if (!this->ldap)
|
|
return;
|
|
|
|
IdentifyInfo *ii = new IdentifyInfo(u, req, this->ldap);
|
|
try
|
|
{
|
|
LDAPQuery id = this->ldap->BindAsAdmin(&this->iinterface);
|
|
this->iinterface.Add(id, ii);
|
|
}
|
|
catch (const LDAPException &ex)
|
|
{
|
|
delete ii;
|
|
Log(this) << ex.GetReason();
|
|
}
|
|
}
|
|
|
|
void OnNickIdentify(User *u) anope_override
|
|
{
|
|
if (email_attribute.empty() || !this->ldap || !u->Account()->HasExt("m_ldap_authentication_dn"))
|
|
return;
|
|
|
|
Anope::string *dn = u->Account()->GetExt<ExtensibleItemClass<Anope::string> *>("m_ldap_authentication_dn");
|
|
if (!dn || dn->empty())
|
|
return;
|
|
|
|
try
|
|
{
|
|
LDAPQuery id = this->ldap->Search(&this->oninterface, *dn, "(" + email_attribute + "=*)");
|
|
this->oninterface.Add(id, u->nick);
|
|
}
|
|
catch (const LDAPException &ex)
|
|
{
|
|
Log(this) << ex.GetReason();
|
|
}
|
|
}
|
|
|
|
void OnNickRegister(NickAlias *na) anope_override
|
|
{
|
|
if (this->disable_register || !this->ldap)
|
|
return;
|
|
|
|
try
|
|
{
|
|
this->ldap->BindAsAdmin(NULL);
|
|
|
|
LDAPMods attributes;
|
|
attributes.resize(4);
|
|
|
|
attributes[0].name = "objectClass";
|
|
attributes[0].values.push_back("top");
|
|
attributes[0].values.push_back(object_class);
|
|
|
|
attributes[1].name = username_attribute;
|
|
attributes[1].values.push_back(na->nick);
|
|
|
|
if (!na->nc->email.empty())
|
|
{
|
|
attributes[2].name = email_attribute;
|
|
attributes[2].values.push_back(na->nc->email);
|
|
}
|
|
|
|
attributes[3].name = this->password_attribute;
|
|
attributes[3].values.push_back(na->nc->pass);
|
|
|
|
Anope::string new_dn = username_attribute + "=" + na->nick + "," + basedn;
|
|
this->ldap->Add(&this->orinterface, new_dn, attributes);
|
|
}
|
|
catch (const LDAPException &ex)
|
|
{
|
|
Log(this) << ex.GetReason();
|
|
}
|
|
}
|
|
};
|
|
|
|
MODULE_INIT(NSIdentifyLDAP)
|