1 /* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
3 * Copyright (C) 2001-2003, Ximian, Inc.
6 #include "test-utils.h"
19 compare_strings (gconstpointer a, gconstpointer b)
21 const char **sa = (const char **)a;
22 const char **sb = (const char **)b;
24 return strcmp (*sa, *sb);
28 get_directory_listing (const char *path)
37 entries = g_ptr_array_new ();
40 while ((dent = readdir (dir))) {
41 if (!strcmp (dent->d_name, ".") ||
42 (!strcmp (dent->d_name, "..") &&
43 !strcmp (path, "./")))
45 escaped = g_markup_escape_text (dent->d_name, -1);
46 g_ptr_array_add (entries, escaped);
51 g_ptr_array_sort (entries, (GCompareFunc)compare_strings);
53 listing = g_string_new ("<html>\r\n");
54 escaped = g_markup_escape_text (strchr (path, '/'), -1);
55 g_string_append_printf (listing, "<head><title>Index of %s</title></head>\r\n", escaped);
56 g_string_append_printf (listing, "<body><h1>Index of %s</h1>\r\n<p>\r\n", escaped);
58 for (i = 0; i < entries->len; i++) {
59 g_string_append_printf (listing, "<a href=\"%s\">%s</a><br>\r\n",
60 (char *)entries->pdata[i],
61 (char *)entries->pdata[i]);
62 g_free (entries->pdata[i]);
64 g_string_append (listing, "</body>\r\n</html>\r\n");
66 g_ptr_array_free (entries, TRUE);
71 do_get (SoupServer *server, SoupMessage *msg, const char *path)
76 if (stat (path, &st) == -1) {
78 soup_message_set_status (msg, SOUP_STATUS_FORBIDDEN);
79 else if (errno == ENOENT)
80 soup_message_set_status (msg, SOUP_STATUS_NOT_FOUND);
82 soup_message_set_status (msg, SOUP_STATUS_INTERNAL_SERVER_ERROR);
86 if (S_ISDIR (st.st_mode)) {
90 slash = strrchr (path, '/');
91 if (!slash || slash[1]) {
94 redir_uri = g_strdup_printf ("%s/", soup_message_get_uri (msg)->path);
95 soup_message_set_redirect (msg, SOUP_STATUS_MOVED_PERMANENTLY,
101 index_path = g_strdup_printf ("%s/index.html", path);
102 if (stat (index_path, &st) != -1) {
103 do_get (server, msg, index_path);
109 listing = get_directory_listing (path);
110 soup_message_set_response (msg, "text/html",
112 listing->str, listing->len);
113 g_string_free (listing, FALSE);
117 if (msg->method == SOUP_METHOD_GET) {
118 GMappedFile *mapping;
121 mapping = g_mapped_file_new (path, FALSE, NULL);
123 soup_message_set_status (msg, SOUP_STATUS_INTERNAL_SERVER_ERROR);
127 buffer = soup_buffer_new_with_owner (g_mapped_file_get_contents (mapping),
128 g_mapped_file_get_length (mapping),
129 mapping, (GDestroyNotify)g_mapped_file_unref);
130 soup_message_body_append_buffer (msg->response_body, buffer);
131 soup_buffer_free (buffer);
132 } else /* msg->method == SOUP_METHOD_HEAD */ {
135 /* We could just use the same code for both GET and
136 * HEAD (soup-message-server-io.c will fix things up).
137 * But we'll optimize and avoid the extra I/O.
139 length = g_strdup_printf ("%lu", (gulong)st.st_size);
140 soup_message_headers_append (msg->response_headers,
141 "Content-Length", length);
145 soup_message_set_status (msg, SOUP_STATUS_OK);
149 do_put (SoupServer *server, SoupMessage *msg, const char *path)
153 gboolean created = TRUE;
155 if (stat (path, &st) != -1) {
156 const char *match = soup_message_headers_get_one (msg->request_headers, "If-None-Match");
157 if (match && !strcmp (match, "*")) {
158 soup_message_set_status (msg, SOUP_STATUS_CONFLICT);
162 if (!S_ISREG (st.st_mode)) {
163 soup_message_set_status (msg, SOUP_STATUS_FORBIDDEN);
170 f = fopen (path, "w");
172 soup_message_set_status (msg, SOUP_STATUS_INTERNAL_SERVER_ERROR);
176 fwrite (msg->request_body->data, 1, msg->request_body->length, f);
179 soup_message_set_status (msg, created ? SOUP_STATUS_CREATED : SOUP_STATUS_OK);
183 server_callback (SoupServer *server, SoupMessage *msg,
184 const char *path, GHashTable *query,
185 SoupClientContext *context, gpointer data)
188 SoupMessageHeadersIter iter;
189 const char *name, *value;
191 g_print ("%s %s HTTP/1.%d\n", msg->method, path,
192 soup_message_get_http_version (msg));
193 soup_message_headers_iter_init (&iter, msg->request_headers);
194 while (soup_message_headers_iter_next (&iter, &name, &value))
195 g_print ("%s: %s\n", name, value);
196 if (msg->request_body->length)
197 g_print ("%s\n", msg->request_body->data);
199 file_path = g_strdup_printf (".%s", path);
201 if (msg->method == SOUP_METHOD_GET || msg->method == SOUP_METHOD_HEAD)
202 do_get (server, msg, file_path);
203 else if (msg->method == SOUP_METHOD_PUT)
204 do_put (server, msg, file_path);
206 soup_message_set_status (msg, SOUP_STATUS_NOT_IMPLEMENTED);
209 g_print (" -> %d %s\n\n", msg->status_code, msg->reason_phrase);
215 /* Exit cleanly on ^C in case we're valgrinding. */
220 main (int argc, char **argv)
223 SoupServer *server, *ssl_server;
225 int port = SOUP_ADDRESS_ANY_PORT;
226 int ssl_port = SOUP_ADDRESS_ANY_PORT;
227 const char *ssl_cert_file = NULL, *ssl_key_file = NULL;
229 signal (SIGINT, quit);
231 while ((opt = getopt (argc, argv, "p:k:c:s:")) != -1) {
234 port = atoi (optarg);
237 ssl_key_file = optarg;
240 ssl_cert_file = optarg;
243 ssl_port = atoi (optarg);
246 g_printerr ("Usage: %s [-p port] [-c ssl-cert-file -k ssl-key-file [-s ssl-port]]\n",
252 server = soup_server_new (SOUP_SERVER_PORT, port,
253 SOUP_SERVER_SERVER_HEADER, "simple-httpd ",
256 g_printerr ("Unable to bind to server port %d\n", port);
259 soup_server_add_handler (server, NULL,
260 server_callback, NULL, NULL);
261 g_print ("\nStarting Server on port %d\n",
262 soup_server_get_port (server));
263 soup_server_run_async (server);
265 if (ssl_cert_file && ssl_key_file) {
266 ssl_server = soup_server_new (
267 SOUP_SERVER_PORT, ssl_port,
268 SOUP_SERVER_SSL_CERT_FILE, ssl_cert_file,
269 SOUP_SERVER_SSL_KEY_FILE, ssl_key_file,
273 g_printerr ("Unable to bind to SSL server port %d\n", ssl_port);
276 soup_server_add_handler (ssl_server, NULL,
277 server_callback, NULL, NULL);
278 g_print ("Starting SSL Server on port %d\n",
279 soup_server_get_port (ssl_server));
280 soup_server_run_async (ssl_server);
283 g_print ("\nWaiting for requests...\n");
285 loop = g_main_loop_new (NULL, TRUE);
286 g_main_loop_run (loop);