]> git.netwichtig.de Git - user/henk/code/inspircd.git/blobdiff - src/modules/extra/m_ziplink.cpp
Fix STARTTLS sending the 670 numeric within the SSL session, not prior to it
[user/henk/code/inspircd.git] / src / modules / extra / m_ziplink.cpp
index 3fb768757cd4c3f8d35b6d668c7192725abd3742..391ba75839a3bd6a7938d7b49c88567e6ff72789 100644 (file)
@@ -2,8 +2,8 @@
  *       | Inspire Internet Relay Chat Daemon |
  *       +------------------------------------+
  *
- *  InspIRCd: (C) 2002-2008 InspIRCd Development Team
- * See: http://www.inspircd.org/wiki/index.php/Credits
+ *  InspIRCd: (C) 2002-2010 InspIRCd Development Team
+ * See: http://wiki.inspircd.org/Credits
  *
  * This program is free but copyrighted software; see
  *            the file COPYING for details.
 
 #include "inspircd.h"
 #include <zlib.h>
-#include "users.h"
-#include "channels.h"
-#include "modules.h"
-#include "socket.h"
-#include "hashcomp.h"
-#include "transport.h"
-
 #include <iostream>
 
 /* $ModDesc: Provides zlib link support for servers */
 /* $LinkerFlags: -lz */
-/* $ModDep: transport.h */
 
 /*
  * ZLIB_BEST_COMPRESSION (9) is used for all sending of data with
@@ -37,7 +29,7 @@ enum izip_status { IZIP_CLOSED = 0, IZIP_OPEN };
 
 /** Represents an zipped connections extra data
  */
-class izip_session : public classbase
+class izip_session
 {
  public:
        z_stream c_stream;      /* compression stream */
@@ -62,85 +54,35 @@ class ModuleZLib : public Module
        unsigned int net_buffer_size;
  public:
 
-       ModuleZLib(InspIRCd* Me)
-               : Module::Module(Me)
+       ModuleZLib()
        {
-               ServerInstance->Modules->PublishInterface("BufferedSocketHook", this);
-
                sessions = new izip_session[ServerInstance->SE->GetMaxFds()];
                for (int i = 0; i < ServerInstance->SE->GetMaxFds(); i++)
                        sessions[i].status = IZIP_CLOSED;
 
                total_out_compressed = total_in_compressed = 0;
                total_out_uncompressed = total_in_uncompressed = 0;
-               Implementation eventlist[] = { I_OnRawSocketConnect, I_OnRawSocketAccept, I_OnRawSocketClose, I_OnRawSocketRead, I_OnRawSocketWrite, I_OnStats, I_OnRequest };
-               ServerInstance->Modules->Attach(eventlist, this, 7);
+               Implementation eventlist[] = { I_OnStats };
+               ServerInstance->Modules->Attach(eventlist, this, 1);
 
                // Allocate a buffer which is used for reading and writing data
                net_buffer_size = ServerInstance->Config->NetBufferSize;
                net_buffer = new char[net_buffer_size];
        }
 
-       virtual ~ModuleZLib()
+       ~ModuleZLib()
        {
-               ServerInstance->Modules->UnpublishInterface("BufferedSocketHook", this);
                delete[] sessions;
                delete[] net_buffer;
        }
 
-       virtual Version GetVersion()
-       {
-               return Version("$Id$", VF_VENDOR, API_VERSION);
-       }
-
-
-       /* Handle BufferedSocketHook API requests */
-       virtual const char* OnRequest(Request* request)
+       Version GetVersion()
        {
-               ISHRequest* ISR = (ISHRequest*)request;
-               if (strcmp("IS_NAME", request->GetId()) == 0)
-               {
-                       /* Return name */
-                       return "zip";
-               }
-               else if (strcmp("IS_HOOK", request->GetId()) == 0)
-               {
-                       /* Attach to an inspsocket */
-                       const char* ret = "OK";
-                       try
-                       {
-                               ret = ServerInstance->Config->AddIOHook((Module*)this, (BufferedSocket*)ISR->Sock) ? "OK" : NULL;
-                       }
-                       catch (ModuleException& e)
-                       {
-                               return NULL;
-                       }
-                       return ret;
-               }
-               else if (strcmp("IS_UNHOOK", request->GetId()) == 0)
-               {
-                       /* Detach from an inspsocket */
-                       return ServerInstance->Config->DelIOHook((BufferedSocket*)ISR->Sock) ? "OK" : NULL;
-               }
-               else if (strcmp("IS_HSDONE", request->GetId()) == 0)
-               {
-                       /* Check for completion of handshake
-                        * (actually, this module doesnt handshake)
-                        */
-                       return "OK";
-               }
-               else if (strcmp("IS_ATTACH", request->GetId()) == 0)
-               {
-                       /* Attach certificate data to the inspsocket
-                        * (this module doesnt do that, either)
-                        */
-                       return NULL;
-               }
-               return NULL;
+               return Version("Provides zlib link support for servers", VF_VENDOR);
        }
 
        /* Handle stats z (misc stats) */
-       virtual int OnStats(char symbol, User* user, string_list &results)
+       ModResult OnStats(char symbol, User* user, string_list &results)
        {
                if (symbol == 'z')
                {
@@ -174,16 +116,20 @@ class ModuleZLib : public Module
                        results.push_back(sn+" 304 "+user->nick+" :ZIPSTATS percentage_of_original_outbound_traffic        = "+outbound_ratio);
                        results.push_back(sn+" 304 "+user->nick+" :ZIPSTATS percentage_of_orignal_inbound_traffic         = "+inbound_ratio);
                        results.push_back(sn+" 304 "+user->nick+" :ZIPSTATS total_size_of_original_traffic        = "+combined_ratio);
-                       return 0;
+                       return MOD_RES_PASSTHRU;
                }
 
-               return 0;
+               return MOD_RES_PASSTHRU;
        }
 
-       virtual void OnRawSocketConnect(int fd)
+       void OnStreamSocketConnect(StreamSocket* user)
        {
-               if ((fd < 0) || (fd > ServerInstance->SE->GetMaxFds() - 1))
-                       return;
+               OnStreamSocketAccept(user, 0, 0);
+       }
+
+       void OnRawSocketAccept(StreamSocket* user, irc::sockets::sockaddrs*, irc::sockets::sockaddrs*)
+       {
+               int fd = user->GetFd();
 
                izip_session* session = &sessions[fd];
 
@@ -217,39 +163,33 @@ class ModuleZLib : public Module
                session->status = IZIP_OPEN;
        }
 
-       virtual void OnRawSocketAccept(int fd, const std::string &ip, int localport)
-       {
-               /* Nothing special needs doing here compared to connect() */
-               OnRawSocketConnect(fd);
-       }
-
-       virtual void OnRawSocketClose(int fd)
+       void OnStreamSocketClose(StreamSocket* user)
        {
+               int fd = user->GetFd();
                CloseSession(&sessions[fd]);
        }
 
-       virtual int OnRawSocketRead(int fd, char* buffer, unsigned int count, int &readresult)
+       int OnStreamSocketRead(StreamSocket* user, std::string& recvq)
        {
+               int fd = user->GetFd();
                /* Find the sockets session */
                izip_session* session = &sessions[fd];
 
                if (session->status == IZIP_CLOSED)
-                       return 0;
+                       return -1;
 
-               if (session->inbuf.length())
-               {
-                       /* Our input buffer is filling up. This is *BAD*.
-                        * We can't return more data than fits into buffer
-                        * (count bytes), so we will generate another read
-                        * event on purpose by *NOT* reading from 'fd' at all
-                        * for now.
-                        */
-                       readresult = 0;
-               }
-               else
+               if (session->inbuf.empty())
                {
                        /* Read read_buffer_size bytes at a time to the buffer (usually 2.5k) */
-                       readresult = read(fd, net_buffer, net_buffer_size);
+                       int readresult = read(fd, net_buffer, net_buffer_size);
+
+                       if (readresult < 0)
+                       {
+                               if (errno == EINTR || errno == EAGAIN)
+                                       return 0;
+                       }
+                       if (readresult <= 0)
+                               return -1;
 
                        total_in_compressed += readresult;
 
@@ -258,10 +198,8 @@ class ModuleZLib : public Module
                }
 
                size_t in_len = session->inbuf.length();
-
-               /* Do we have anything to do? */
-               if (in_len <= 0)
-                       return 0;
+               char* buffer = ServerInstance->GetReadBuffer();
+               int count = ServerInstance->Config->NetBufferSize;
 
                /* Prepare decompression */
                session->d_stream.next_in = (Bytef *)session->inbuf.c_str();
@@ -308,8 +246,7 @@ class ModuleZLib : public Module
                }
                if (ret != Z_OK)
                {
-                       readresult = 0;
-                       return 0;
+                       return -1;
                }
 
                /* Update the inbut buffer */
@@ -321,24 +258,18 @@ class ModuleZLib : public Module
                total_in_uncompressed += uncompressed_length;
 
                /* Null-terminate the buffer -- this doesnt harm binary data */
-               buffer[uncompressed_length] = 0;
-
-               /* Set the read size to the correct total size */
-               readresult = uncompressed_length;
-
+               recvq.append(buffer, uncompressed_length);
                return 1;
        }
 
-       virtual int OnRawSocketWrite(int fd, const char* buffer, int count)
+       int OnStreamSocketWrite(StreamSocket* user, std::string& sendq)
        {
+               int fd = user->GetFd();
                izip_session* session = &sessions[fd];
 
-               if (!count)     /* Nothing to do! */
-                       return 0;
-
                if(session->status != IZIP_OPEN)
                        /* Seriously, wtf? */
-                       return 0;
+                       return -1;
 
                int ret;
 
@@ -349,8 +280,8 @@ class ModuleZLib : public Module
                do
                {
                        /* Prepare compression */
-                       session->c_stream.next_in = (Bytef*)buffer + offset;
-                       session->c_stream.avail_in = count - offset;
+                       session->c_stream.next_in = (Bytef*)sendq.data() + offset;
+                       session->c_stream.avail_in = sendq.length() - offset;
 
                        session->c_stream.next_out = (Bytef*)net_buffer;
                        session->c_stream.avail_out = net_buffer_size;
@@ -384,7 +315,7 @@ class ModuleZLib : public Module
 
                        /* Space before - space after stuff was added to this */
                        unsigned int compressed = net_buffer_size - session->c_stream.avail_out;
-                       unsigned int uncompressed = count - session->c_stream.avail_in;
+                       unsigned int uncompressed = sendq.length() - session->c_stream.avail_in;
 
                        /* Make it skip the data which was compressed already */
                        offset += uncompressed;
@@ -410,14 +341,11 @@ class ModuleZLib : public Module
                        else
                        {
                                session->outbuf.clear();
-                               return 0;
+                               return -1;
                        }
                }
 
-               /* ALL LIES the lot of it, we havent really written
-                * this amount, but the layer above doesnt need to know.
-                */
-               return count;
+               return 1;
        }
 
        void Error(izip_session* session, const std::string &text)