/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
/*
* Copyright (C) 2001-2003, Ximian, Inc.
*/
#include <ctype.h>
#include <fcntl.h>
#include <errno.h>
#include <signal.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/stat.h>
#include <unistd.h>
#include <glib.h>
#include <libsoup/soup-address.h>
#include <libsoup/soup-message.h>
#include <libsoup/soup-server.h>
#include <libsoup/soup-server-message.h>
static void
print_header (gpointer name, gpointer value, gpointer data)
{
printf ("%s: %s\n", (char *)name, (char *)value);
}
static void
server_callback (SoupServerContext *context, SoupMessage *msg, gpointer data)
{
char *path, *path_to_open, *slash;
SoupMethodId method;
struct stat st;
int fd;
path = soup_uri_to_string (soup_message_get_uri (msg), TRUE);
printf ("%s %s HTTP/1.%d\n", msg->method, path,
soup_message_get_http_version (msg));
soup_message_foreach_header (msg->request_headers, print_header, NULL);
if (msg->request.length)
printf ("%.*s\n", msg->request.length, msg->request.body);
method = soup_method_get_id (msg->method);
if (method != SOUP_METHOD_ID_GET && method != SOUP_METHOD_ID_HEAD) {
soup_message_set_status (msg, SOUP_STATUS_NOT_IMPLEMENTED);
goto DONE;
}
if (path) {
if (*path != '/') {
soup_message_set_status (msg, SOUP_STATUS_BAD_REQUEST);
goto DONE;
}
} else
path = g_strdup ("");
path_to_open = g_strdup_printf (".%s", path);
AGAIN:
if (stat (path_to_open, &st) == -1) {
g_free (path_to_open);
if (errno == EPERM)
soup_message_set_status (msg, SOUP_STATUS_FORBIDDEN);
else if (errno == ENOENT)
soup_message_set_status (msg, SOUP_STATUS_NOT_FOUND);
else
soup_message_set_status (msg, SOUP_STATUS_INTERNAL_SERVER_ERROR);
goto DONE;
}
if (S_ISDIR (st.st_mode)) {
slash = strrchr (path_to_open, '/');
if (!slash || slash[1]) {
char *uri, *redir_uri;
uri = soup_uri_to_string (soup_message_get_uri (msg), FALSE);
redir_uri = g_strdup_printf ("%s/", uri);
soup_message_add_header (msg->response_headers,
"Location", redir_uri);
soup_message_set_status (msg, SOUP_STATUS_MOVED_PERMANENTLY);
g_free (redir_uri);
g_free (uri);
g_free (path_to_open);
goto DONE;
}
g_free (path_to_open);
path_to_open = g_strdup_printf (".%s/index.html", path);
goto AGAIN;
}
fd = open (path_to_open, O_RDONLY);
g_free (path_to_open);
if (fd == -1) {
soup_message_set_status (msg, SOUP_STATUS_INTERNAL_SERVER_ERROR);
goto DONE;
}
msg->response.owner = SOUP_BUFFER_SYSTEM_OWNED;
msg->response.length = st.st_size;
if (method == SOUP_METHOD_ID_GET) {
msg->response.body = g_malloc (msg->response.length);
read (fd, msg->response.body, msg->response.length);
} else /* method == SOUP_METHOD_ID_HEAD */ {
/* SoupServer will ignore response.body and only use
* response.length when responding to HEAD, so we
* could just use the same code for both GET and HEAD.
* But we'll optimize and avoid the extra malloc.
*/
msg->response.body = NULL;
}
close (fd);
soup_message_set_status (msg, SOUP_STATUS_OK);
DONE:
g_free (path);
printf (" -> %d %s\n\n", msg->status_code, msg->reason_phrase);
}
static void
quit (int sig)
{
/* Exit cleanly on ^C in case we're valgrinding. */
exit (0);
}
int
main (int argc, char **argv)
{
GMainLoop *loop;
SoupServer *server, *ssl_server;
int opt;
int port = SOUP_ADDRESS_ANY_PORT;
int ssl_port = SOUP_ADDRESS_ANY_PORT;
const char *ssl_cert_file = NULL, *ssl_key_file = NULL;
g_type_init ();
g_thread_init (NULL);
signal (SIGINT, quit);
while ((opt = getopt (argc, argv, "p:k:c:s:")) != -1) {
switch (opt) {
case 'p':
port = atoi (optarg);
break;
case 'k':
ssl_key_file = optarg;
break;
case 'c':
ssl_cert_file = optarg;
break;
case 's':
ssl_port = atoi (optarg);
break;
default:
fprintf (stderr, "Usage: %s [-p port] [-c ssl-cert-file -k ssl-key-file [-s ssl-port]]\n",
argv[0]);
exit (1);
}
}
server = soup_server_new (SOUP_SERVER_PORT, port,
NULL);
if (!server) {
fprintf (stderr, "Unable to bind to server port %d\n", port);
exit (1);
}
soup_server_add_handler (server, NULL, NULL,
server_callback, NULL, NULL);
printf ("\nStarting Server on port %d\n",
soup_server_get_port (server));
soup_server_run_async (server);
if (ssl_cert_file && ssl_key_file) {
ssl_server = soup_server_new (
SOUP_SERVER_PORT, ssl_port,
SOUP_SERVER_SSL_CERT_FILE, ssl_cert_file,
SOUP_SERVER_SSL_KEY_FILE, ssl_key_file,
NULL);
if (!ssl_server) {
fprintf (stderr, "Unable to bind to SSL server port %d\n", ssl_port);
exit (1);
}
soup_server_add_handler (ssl_server, NULL, NULL,
server_callback, NULL, NULL);
printf ("Starting SSL Server on port %d\n",
soup_server_get_port (ssl_server));
soup_server_run_async (ssl_server);
}
printf ("\nWaiting for requests...\n");
loop = g_main_loop_new (NULL, TRUE);
g_main_loop_run (loop);
return 0;
}
syntax highlighted by Code2HTML, v. 0.9.1