1 /* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
3 * Copyright (C) 2001-2003, Ximian, Inc.
23 #include <libsoup/soup.h>
32 free_mapping (gpointer data)
34 struct mapping *mapping = data;
35 munmap (mapping->start, mapping->length);
36 g_slice_free (struct mapping, mapping);
41 compare_strings (gconstpointer a, gconstpointer b)
43 const char **sa = (const char **)a;
44 const char **sb = (const char **)b;
46 return strcmp (*sa, *sb);
50 get_directory_listing (const char *path)
59 entries = g_ptr_array_new ();
62 while ((dent = readdir (dir))) {
63 if (!strcmp (dent->d_name, ".") ||
64 (!strcmp (dent->d_name, "..") &&
65 !strcmp (path, "./")))
67 escaped = g_markup_escape_text (dent->d_name, -1);
68 g_ptr_array_add (entries, escaped);
73 g_ptr_array_sort (entries, (GCompareFunc)compare_strings);
75 listing = g_string_new ("<html>\r\n");
76 escaped = g_markup_escape_text (strchr (path, '/'), -1);
77 g_string_append_printf (listing, "<head><title>Index of %s</title></head>\r\n", escaped);
78 g_string_append_printf (listing, "<body><h1>Index of %s</h1>\r\n<p>\r\n", escaped);
80 for (i = 0; i < entries->len; i++) {
81 g_string_append_printf (listing, "<a href=\"%s\">%s</a><br>\r\n",
82 (char *)entries->pdata[i],
83 (char *)entries->pdata[i]);
84 g_free (entries->pdata[i]);
86 g_string_append (listing, "</body>\r\n</html>\r\n");
88 g_ptr_array_free (entries, TRUE);
93 do_get (SoupServer *server, SoupMessage *msg, const char *path)
99 if (stat (path, &st) == -1) {
101 soup_message_set_status (msg, SOUP_STATUS_FORBIDDEN);
102 else if (errno == ENOENT)
103 soup_message_set_status (msg, SOUP_STATUS_NOT_FOUND);
105 soup_message_set_status (msg, SOUP_STATUS_INTERNAL_SERVER_ERROR);
109 if (S_ISDIR (st.st_mode)) {
113 slash = strrchr (path, '/');
114 if (!slash || slash[1]) {
115 char *uri, *redir_uri;
117 uri = soup_uri_to_string (soup_message_get_uri (msg), FALSE);
118 redir_uri = g_strdup_printf ("%s/", uri);
119 soup_message_headers_append (msg->response_headers,
120 "Location", redir_uri);
121 soup_message_set_status (msg, SOUP_STATUS_MOVED_PERMANENTLY);
127 index_path = g_strdup_printf ("%s/index.html", path);
128 if (stat (index_path, &st) != -1) {
129 do_get (server, msg, index_path);
134 listing = get_directory_listing (path);
135 soup_message_set_response (msg, "text/html",
137 listing->str, listing->len);
138 g_string_free (listing, FALSE);
142 fd = open (path, O_RDONLY);
144 soup_message_set_status (msg, SOUP_STATUS_INTERNAL_SERVER_ERROR);
148 if (msg->method == SOUP_METHOD_GET) {
150 struct mapping *mapping = g_slice_new (struct mapping);
153 mapping->start = mmap (NULL, st.st_size, PROT_READ,
155 mapping->length = st.st_size;
156 buffer = soup_buffer_new_with_owner (mapping->start,
158 mapping, free_mapping);
159 soup_message_body_append_buffer (msg->response_body, buffer);
160 soup_buffer_free (buffer);
164 buf = g_malloc (st.st_size);
165 read (fd, buf, st.st_size);
167 soup_message_body_append (msg->response_body, SOUP_MEMORY_TAKE,
170 } else /* msg->method == SOUP_METHOD_HEAD */ {
173 /* We could just use the same code for both GET and
174 * HEAD. But we'll optimize and avoid the extra
177 length = g_strdup_printf ("%lu", (gulong)st.st_size);
178 soup_message_headers_append (msg->response_headers,
179 "Content-Length", length);
183 soup_message_set_status (msg, SOUP_STATUS_OK);
187 do_put (SoupServer *server, SoupMessage *msg, const char *path)
191 gboolean created = TRUE;
193 if (stat (path, &st) != -1) {
194 const char *match = soup_message_headers_get_one (msg->request_headers, "If-None-Match");
195 if (match && !strcmp (match, "*")) {
196 soup_message_set_status (msg, SOUP_STATUS_CONFLICT);
200 if (!S_ISREG (st.st_mode)) {
201 soup_message_set_status (msg, SOUP_STATUS_FORBIDDEN);
208 f = fopen (path, "w");
210 soup_message_set_status (msg, SOUP_STATUS_INTERNAL_SERVER_ERROR);
214 fwrite (msg->request_body->data, 1, msg->request_body->length, f);
217 soup_message_set_status (msg, created ? SOUP_STATUS_CREATED : SOUP_STATUS_OK);
221 server_callback (SoupServer *server, SoupMessage *msg,
222 const char *path, GHashTable *query,
223 SoupClientContext *context, gpointer data)
226 SoupMessageHeadersIter iter;
227 const char *name, *value;
229 printf ("%s %s HTTP/1.%d\n", msg->method, path,
230 soup_message_get_http_version (msg));
231 soup_message_headers_iter_init (&iter, msg->request_headers);
232 while (soup_message_headers_iter_next (&iter, &name, &value))
233 printf ("%s: %s\n", name, value);
234 if (msg->request_body->length)
235 printf ("%s\n", msg->request_body->data);
237 file_path = g_strdup_printf (".%s", path);
239 if (msg->method == SOUP_METHOD_GET || msg->method == SOUP_METHOD_HEAD)
240 do_get (server, msg, file_path);
241 else if (msg->method == SOUP_METHOD_PUT)
242 do_put (server, msg, file_path);
244 soup_message_set_status (msg, SOUP_STATUS_NOT_IMPLEMENTED);
247 printf (" -> %d %s\n\n", msg->status_code, msg->reason_phrase);
253 /* Exit cleanly on ^C in case we're valgrinding. */
258 main (int argc, char **argv)
261 SoupServer *server, *ssl_server;
263 int port = SOUP_ADDRESS_ANY_PORT;
264 int ssl_port = SOUP_ADDRESS_ANY_PORT;
265 const char *ssl_cert_file = NULL, *ssl_key_file = NULL;
267 g_thread_init (NULL);
269 signal (SIGINT, quit);
271 while ((opt = getopt (argc, argv, "p:k:c:s:")) != -1) {
274 port = atoi (optarg);
277 ssl_key_file = optarg;
280 ssl_cert_file = optarg;
283 ssl_port = atoi (optarg);
286 fprintf (stderr, "Usage: %s [-p port] [-c ssl-cert-file -k ssl-key-file [-s ssl-port]]\n",
292 server = soup_server_new (SOUP_SERVER_PORT, port,
293 SOUP_SERVER_SERVER_HEADER, "simple-httpd ",
296 fprintf (stderr, "Unable to bind to server port %d\n", port);
299 soup_server_add_handler (server, NULL,
300 server_callback, NULL, NULL);
301 printf ("\nStarting Server on port %d\n",
302 soup_server_get_port (server));
303 soup_server_run_async (server);
305 if (ssl_cert_file && ssl_key_file) {
306 ssl_server = soup_server_new (
307 SOUP_SERVER_PORT, ssl_port,
308 SOUP_SERVER_SSL_CERT_FILE, ssl_cert_file,
309 SOUP_SERVER_SSL_KEY_FILE, ssl_key_file,
313 fprintf (stderr, "Unable to bind to SSL server port %d\n", ssl_port);
316 soup_server_add_handler (ssl_server, NULL,
317 server_callback, NULL, NULL);
318 printf ("Starting SSL Server on port %d\n",
319 soup_server_get_port (ssl_server));
320 soup_server_run_async (ssl_server);
323 printf ("\nWaiting for requests...\n");
325 loop = g_main_loop_new (NULL, TRUE);
326 g_main_loop_run (loop);