/* Debug page module for HTTP server.
 *
 * IRC Services is copyright (c) 1996-2007 Andrew Church.
 *     E-mail: <achurch@achurch.org>
 * Parts written by Andrew Kempe and others.
 * This program is free but copyrighted software; see the file COPYING for
 * details.
 */

#include "services.h"
#include "modules.h"
#include "conffile.h"
#include "http.h"

/*************************************************************************/

static Module *module;
static Module *module_httpd;

static char *DebugURL;

/*************************************************************************/
/*************************** Request callback ****************************/
/*************************************************************************/

static int do_request(Client *c, int *close_ptr)
{
    /* Check whether this URL belongs to us.  If not, pass control on to
     * the next callback function.
     */
    if (strcmp(c->url, DebugURL) != 0)
	return 0;


    /* Send initial "200 OK" line and Date: header */
    http_send_response(c, HTTP_S_OK);

    /* If not a HEAD request, indicate that we will close the connection
     * after this request (because we don't send a Content-Length:
     * header).  If the request is a HEAD request, the blank line after
     * the headers will signal the end of the response, so we can leave
     * the connection open for further requests (keepalive).
     */
    if (c->method != METHOD_HEAD)
	sockprintf(c->socket, "Connection: close\r\n");

    /* Send Content-Type: header and end header portion of response */
    sockprintf(c->socket, "Content-Type: text/plain\r\n\r\n");


    /* Now send the body part of the response for non-HEAD requests.
     *
     * RFC2616 9.4: Server MUST NOT return a message-body in the response
     *              to a HEAD request.
     */
    if (c->method != METHOD_HEAD) {
	int i;

	/* Write data to socket */
	sockprintf(c->socket, "address: %s\n", c->address);
	sockprintf(c->socket, "request_len: %d\n", c->request_len);
	sockprintf(c->socket, "version_major: %d\n", c->version_major);
	sockprintf(c->socket, "version_minor: %d\n", c->version_minor);
	sockprintf(c->socket, "method: %d\n", c->method);
	sockprintf(c->socket, "url: %s\n", c->url);
	sockprintf(c->socket, "data_len: %d\n", c->data_len);
	sockprintf(c->socket, "headers_count: %d\n", c->headers_count);
	ARRAY_FOREACH (i, c->headers)
	    sockprintf(c->socket, "headers[%d]: %s: %s\n", i, c->headers[i],
		       c->headers[i] + strlen(c->headers[i]) + 1);
	sockprintf(c->socket, "variables_count: %d\n", c->variables_count);
	ARRAY_FOREACH (i, c->variables)
	    sockprintf(c->socket, "variables[%d]: %s: %s\n", i,c->variables[i],
		       c->variables[i] + strlen(c->variables[i]) + 1);

	/* We did not specify a Content-Length: header, so we must close
	 * the connection to signal end-of-data to the client.  However,
	 * we MUST NOT call disconn() directly as that could lead to
	 * use of invalid pointers in the main HTTP server module.
	 * Instead, we set `close_ptr' nonzer, which tells the server to
	 * close the connection when control returns to it.
	 */
	*close_ptr = 1;
    }
    /* Note that we MUST NOT explicitly set `close_ptr' to zero for HEAD
     * requests, or any other request for which we can keep the connection
     * alive; the client may be an old (HTTP/1.0) client that can't handle
     * keepalive, or it may have explicitly requested the connection be
     * closed (e.g. with a "Connection: close" header).
     */

    /* URL was handled by this module; terminate callback chain. */
    return 1;
}

/*************************************************************************/
/***************************** Module stuff ******************************/
/*************************************************************************/

const int32 module_version = MODULE_VERSION_CODE;

ConfigDirective module_config[] = {
    { "DebugURL",         { { CD_STRING, CF_DIRREQ, &DebugURL } } },
    { NULL }
};

/*************************************************************************/

int init_module(Module *module_)
{
    module = module_;
    module_httpd = NULL;

    module_httpd = find_module("httpd/main");
    if (!module_httpd) {
	module_log("Main httpd module not loaded");
	exit_module(0);
	return 0;
    }
    use_module(module_httpd);

    if (!add_callback(module_httpd, "request", do_request)) {
	module_log("Unable to add callback");
	exit_module(0);
	return 0;
    }

    return 1;
}

/*************************************************************************/

int exit_module(int shutdown_unused)
{
#ifdef CLEAN_COMPILE
    shutdown_unused = shutdown_unused;
#endif

    if (module_httpd) {
	remove_callback(module_httpd, "request", do_request);
	unuse_module(module_httpd);
	module_httpd = NULL;
    }

    return 1;
}

/*************************************************************************/


syntax highlighted by Code2HTML, v. 0.9.1