/* +------------------------------------+ * | Inspire Internet Relay Chat Daemon | * +------------------------------------+ * * InspIRCd: (C) 2002-2008 InspIRCd Development Team * See: http://www.inspircd.org/wiki/index.php/Credits * * This program is free but copyrighted software; see * the file COPYING for details. * * --------------------------------------------------- */ #include "inspircd.h" #include "m_cap.h" #include "account.h" /* $ModDesc: Provides support for IRC Authentication Layer (aka: atheme SASL) via AUTHENTICATE. */ enum SaslState { SASL_INIT, SASL_COMM, SASL_DONE }; enum SaslResult { SASL_OK, SASL_FAIL, SASL_ABORT }; /** * Tracks SASL authentication state like charybdis does. --nenolod */ class SaslAuthenticator : public classbase { private: InspIRCd *ServerInstance; Module *Creator; std::string agent; User *user; SaslState state; SaslResult result; bool state_announced; public: SaslAuthenticator(User *user_, std::string method, InspIRCd *instance, Module *ctor) : ServerInstance(instance), Creator(ctor), user(user_), state(SASL_INIT), state_announced(false) { this->user->Extend("sasl_authenticator", this); std::deque<std::string> params; params.push_back("*"); params.push_back("SASL"); params.push_back(user->uuid); params.push_back("*"); params.push_back("S"); params.push_back(method); ServerInstance->PI->SendEncapsulatedData(params); } SaslResult GetSaslResult(std::string &result_) { if (result_ == "F") return SASL_FAIL; if (result_ == "A") return SASL_ABORT; return SASL_OK; } /* checks for and deals with a state change. */ SaslState ProcessInboundMessage(std::deque<std::string> &msg) { switch (this->state) { case SASL_INIT: this->agent = msg[2]; this->user->Write("AUTHENTICATE %s", msg[5].c_str()); this->state = SASL_COMM; break; case SASL_COMM: if (msg[2] != this->agent) return this->state; if (msg[4] != "D") this->user->Write("AUTHENTICATE %s", msg[5].c_str()); else { this->state = SASL_DONE; this->result = this->GetSaslResult(msg[5]); } break; case SASL_DONE: break; default: ServerInstance->Logs->Log("m_sasl", DEFAULT, "WTF: SaslState is not a known state (%d)", this->state); break; } return this->state; } void Abort(void) { this->state = SASL_DONE; this->result = SASL_ABORT; } bool SendClientMessage(const std::vector<std::string>& parameters) { if (this->state != SASL_COMM) return true; std::deque<std::string> params; params.push_back("*"); params.push_back("SASL"); params.push_back(this->user->uuid); params.push_back(this->agent); params.push_back("C"); params.insert(params.end(), parameters.begin(), parameters.end()); ServerInstance->PI->SendEncapsulatedData(params); if (parameters[0][0] == '*') { this->Abort(); return false; } return true; } void AnnounceState(void) { if (this->state_announced) return; switch (this->result) { case SASL_OK: this->user->WriteNumeric(903, "%s :SASL authentication successful", this->user->nick.c_str()); break; case SASL_ABORT: this->user->WriteNumeric(906, "%s :SASL authentication aborted", this->user->nick.c_str()); break; case SASL_FAIL: this->user->WriteNumeric(904, "%s :SASL authentication failed", this->user->nick.c_str()); break; default: break; } this->state_announced = true; } ~SaslAuthenticator() { this->user->Shrink("sasl_authenticator"); this->AnnounceState(); } }; class CommandAuthenticate : public Command { Module* Creator; public: CommandAuthenticate (InspIRCd* Instance, Module* creator) : Command(Instance,"AUTHENTICATE", 0, 1, true), Creator(creator) { this->source = "m_sasl.so"; } CmdResult Handle (const std::vector<std::string>& parameters, User *user) { /* Only allow AUTHENTICATE on unregistered clients */ if (user->registered != REG_ALL) { if (!user->GetExt("sasl")) return CMD_FAILURE; SaslAuthenticator *sasl; if (!(user->GetExt("sasl_authenticator", sasl))) sasl = new SaslAuthenticator(user, parameters[0], ServerInstance, Creator); else if (sasl->SendClientMessage(parameters) == false) // IAL abort extension --nenolod delete sasl; } return CMD_FAILURE; } }; class ModuleSASL : public Module { CommandAuthenticate* sasl; public: ModuleSASL(InspIRCd* Me) : Module(Me) { Implementation eventlist[] = { I_OnEvent, I_OnUserRegister, I_OnPostConnect, I_OnUserDisconnect, I_OnCleanup }; ServerInstance->Modules->Attach(eventlist, this, 5); sasl = new CommandAuthenticate(ServerInstance, this); ServerInstance->AddCommand(sasl); if (!ServerInstance->Modules->Find("m_services_account.so") || !ServerInstance->Modules->Find("m_cap.so")) ServerInstance->Logs->Log("m_sasl", DEFAULT, "WARNING: m_services_account.so and m_cap.so are not loaded! m_sasl.so will NOT function correctly until these two modules are loaded!"); } virtual int OnUserRegister(User *user) { SaslAuthenticator *sasl_; if (user->GetExt("sasl_authenticator", sasl_)) { sasl_->Abort(); delete sasl_; user->Shrink("sasl_authenticator"); } return 0; } virtual void OnCleanup(int target_type, void *item) { if (target_type == TYPE_USER) OnUserDisconnect((User*)item); } virtual void OnUserDisconnect(User *user) { SaslAuthenticator *sasl_; if (user->GetExt("sasl_authenticator", sasl_)) { delete sasl_; user->Shrink("sasl_authenticator"); } } virtual void OnPostConnect(User* user) { if (!IS_LOCAL(user)) return; std::string* str = NULL; if (user->GetExt("accountname", str)) ServerInstance->PI->SendMetaData(user, TYPE_USER, "accountname", *str); return; } virtual ~ModuleSASL() { } virtual Version GetVersion() { return Version("$Id$",VF_VENDOR,API_VERSION); } virtual void OnEvent(Event *ev) { GenericCapHandler(ev, "sasl", "sasl"); if (ev->GetEventID() == "encap_received") { std::deque<std::string>* parameters = (std::deque<std::string>*)ev->GetData(); if ((*parameters)[1] != "SASL") return; User* target = ServerInstance->FindNick((*parameters)[3]); if (!target) { ServerInstance->Logs->Log("m_sasl", DEBUG,"User not found in sasl ENCAP event: %s", (*parameters)[3].c_str()); return; } SaslAuthenticator *sasl_; if (!target->GetExt("sasl_authenticator", sasl_)) return; SaslState state = sasl_->ProcessInboundMessage(*parameters); if (state == SASL_DONE) { delete sasl_; target->Shrink("sasl"); } } } }; MODULE_INIT(ModuleSASL)