diff options
Diffstat (limited to 'modules/extra/m_xmlrpc.cpp')
-rw-r--r-- | modules/extra/m_xmlrpc.cpp | 270 |
1 files changed, 270 insertions, 0 deletions
diff --git a/modules/extra/m_xmlrpc.cpp b/modules/extra/m_xmlrpc.cpp new file mode 100644 index 000000000..f71a260d0 --- /dev/null +++ b/modules/extra/m_xmlrpc.cpp @@ -0,0 +1,270 @@ +#include "module.h" +#include "ssl.h" +#include "xmlrpc.h" + +class MyXMLRPCClientSocket : public XMLRPCClientSocket +{ + public: + MyXMLRPCClientSocket(XMLRPCListenSocket *ls, int fd, const sockaddrs &addr) : XMLRPCClientSocket(ls, fd, addr) + { + if (ls->username.empty() && ls->password.empty()) + this->logged_in = true; + } + + bool Read(const Anope::string &message) + { + this->query += message; + Log(LOG_DEBUG) << "m_xmlrpc: " << message; + + if (message.find("</methodCall>") != Anope::string::npos) + { + this->HandleMessage(); + this->query.clear(); + } + return true; + } + + bool GetData(Anope::string &tag, Anope::string &data) + { + if (this->query.empty()) + return false; + + Anope::string prev, cur; + bool istag; + + do + { + prev = cur; + cur.clear(); + + int len = 0; + istag = false; + + if (this->query[0] == '<') + { + len = this->query.find_first_of('>'); + istag = true; + } + else if (this->query[0] != '>') + { + len = this->query.find_first_of('<'); + } + + if (len) + { + if (istag) + { + cur = this->query.substr(1, len - 1); + this->query.erase(0, len + 1); + while (!this->query.empty() && this->query[0] == ' ') + this->query.erase(this->query.begin()); + } + else + { + cur = this->query.substr(0,len); + this->query.erase(0, len); + } + } + } + while (istag && !this->query.empty()); + + tag = prev; + data = cur; + return !istag && !data.empty(); + } + + void HandleMessage(); +}; + +class MyXMLRPCListenSocket : public XMLRPCListenSocket +{ + public: + MyXMLRPCListenSocket(const Anope::string &bindip, int port, bool ipv6, const Anope::string &u, const Anope::string &p, const std::vector<Anope::string> &a) : XMLRPCListenSocket(bindip, port, ipv6, u, p, a) { } + + ClientSocket *OnAccept(int fd, const sockaddrs &addr) + { + MyXMLRPCClientSocket *socket = new MyXMLRPCClientSocket(this, fd, addr); + if (std::find(this->allowed.begin(), this->allowed.end(), addr.addr()) != this->allowed.end()) + Log() << "m_xmlrpc: Accepted connection " << fd << " from " << addr.addr(); + else + { + Log() << "m_xmlrpc: Dropping disallowed connection " << fd << " from " << addr.addr(); + socket->SetFlag(SF_DEAD); + } + return socket; + } +}; + +class MyXMLRPCServiceInterface : public XMLRPCServiceInterface +{ + std::deque<XMLRPCEvent *> events; + + public: + MyXMLRPCServiceInterface(Module *creator, const Anope::string &sname) : XMLRPCServiceInterface(creator, sname) { } + + void Register(XMLRPCEvent *event) + { + this->events.push_back(event); + } + + void Unregister(XMLRPCEvent *event) + { + std::deque<XMLRPCEvent *>::iterator it = std::find(this->events.begin(), this->events.end(), event); + + if (it != this->events.end()) + this->events.erase(it); + } + + void Reply(XMLRPCClientSocket *source, XMLRPCRequest *request) + { + if (!request->id.empty()) + request->reply("id", request->id); + + Anope::string reply = "<?xml version=\"1.0\" encoding=\"iso-8859-1\"?>\n<methodCall>\n<methodName>" + request->name + "</methodName>\n<params>\n<param>\n<value>\n<struct>\n"; + for (std::map<Anope::string, Anope::string>::const_iterator it = request->get_replies().begin(); it != request->get_replies().end(); ++it) + { + reply += "<member>\n<name>" + it->first + "</name>\n<value>\n<string>" + this->Sanitize(it->second) + "</string>\n</value>\n</member>\n"; + } + reply += "</struct>\n</value>\n</param>\n</params>\n</methodCall>"; + + source->Write(reply); + } + + Anope::string Sanitize(const Anope::string &string) + { + static struct special_chars + { + Anope::string character; + Anope::string replace; + + special_chars(const Anope::string &c, const Anope::string &r) : character(c), replace(r) { } + } + special[] = { + special_chars("&", "&"), + special_chars("\"", """), + special_chars("<", "<"), + special_chars(">", "&qt;"), + special_chars("'", "'"), + special_chars("\n", "
"), + special_chars("\002", ""), // bold + special_chars("\003", ""), // color + special_chars("\35", ""), // italics + special_chars("\031", ""), // underline + special_chars("\26", ""), // reverses + special_chars("", "") + }; + + Anope::string ret = string; + for (int i = 0; special[i].character.empty() == false; ++i) + ret = ret.replace_all_cs(special[i].character, special[i].replace); + return ret; + } + + void RunXMLRPC(XMLRPCClientSocket *source, XMLRPCRequest *request) + { + for (unsigned i = 0; i < this->events.size(); ++i) + { + XMLRPCEvent *e = this->events[i]; + + e->Run(this, source, request); + if (!request->get_replies().empty()) + this->Reply(source, request); + } + } +}; + +class ModuleXMLRPC; +static ModuleXMLRPC *me; +class ModuleXMLRPC : public Module +{ + std::vector<MyXMLRPCListenSocket *> listen_sockets; + service_reference<SSLService> sslref; + + public: + MyXMLRPCServiceInterface xmlrpcinterface; + + ModuleXMLRPC(const Anope::string &modname, const Anope::string &creator) : Module(modname, creator), sslref("ssl"), xmlrpcinterface(this, "xmlrpc") + { + me = this; + + OnReload(false); + + ModuleManager::RegisterService(&this->xmlrpcinterface); + + Implementation i[] = { I_OnReload }; + ModuleManager::Attach(i, this, 1); + } + + ~ModuleXMLRPC() + { + for (unsigned i = 0; i < this->listen_sockets.size(); ++i) + delete this->listen_sockets[i]; + this->listen_sockets.clear(); + } + + void OnReload(bool) + { + ConfigReader config; + + for (unsigned i = 0; i < this->listen_sockets.size(); ++i) + delete this->listen_sockets[i]; + this->listen_sockets.clear(); + + for (int i = 0; i < config.Enumerate("m_xmlrpc"); ++i) + { + Anope::string bindip = config.ReadValue("m_xmlrpc", "bindip", "0.0.0.0", i); + int port = config.ReadInteger("m_xmlrpc", "port", 0, i); + bool ipv6 = config.ReadFlag("m_xmlrpc", "ipv6", "no", i); + bool ssl = config.ReadFlag("m_xmlrpc", "ssl", "no", i); + Anope::string allowed = config.ReadValue("m_xmlrpc", "allowed", "127.0.0.1", i); + Anope::string username = config.ReadValue("m_xmlrpc", "username", "", i); + Anope::string password = config.ReadValue("m_xmlrpc", "password", "", i); + std::vector<Anope::string> allowed_vector = BuildStringVector(allowed, ' '); + + if (bindip.empty() || port < 1) + continue; + + if (ssl && !sslref) + { + Log() << "m_xmlrpc: Could not enable SSL, is m_ssl loaded?"; + ssl = false; + } + + try + { + MyXMLRPCListenSocket *xmls = new MyXMLRPCListenSocket(bindip, port, ipv6, username, password, allowed_vector); + if (ssl) + { + sslref->Init(xmls); + } + this->listen_sockets.push_back(xmls); + } + catch (const SocketException &ex) + { + Log() << "m_xmlrpc " << ex.GetReason(); + } + } + } +}; + +void MyXMLRPCClientSocket::HandleMessage() +{ + Anope::string name, data; + XMLRPCRequest request; + + while (this->GetData(name, data)) + { + if (name == "methodName") + request.name = data; + else if (name == "id") + request.id = data; + else if (name == "string") + request.data.push_back(data); + } + + if (request.name == "login" || this->logged_in) + me->xmlrpcinterface.RunXMLRPC(this, &request); +} + +MODULE_INIT(ModuleXMLRPC) + |