1 /* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
3 * Copyright (C) 2001-2003, Ximian, Inc.
22 #include <libsoup/soup.h>
31 free_mapping (gpointer data)
33 struct mapping *mapping = data;
34 munmap (mapping->start, mapping->length);
35 g_slice_free (struct mapping, mapping);
40 do_get (SoupServer *server, SoupMessage *msg, const char *path)
46 if (stat (path, &st) == -1) {
48 soup_message_set_status (msg, SOUP_STATUS_FORBIDDEN);
49 else if (errno == ENOENT)
50 soup_message_set_status (msg, SOUP_STATUS_NOT_FOUND);
52 soup_message_set_status (msg, SOUP_STATUS_INTERNAL_SERVER_ERROR);
56 if (S_ISDIR (st.st_mode)) {
59 slash = strrchr (path, '/');
60 if (!slash || slash[1]) {
61 char *uri, *redir_uri;
63 uri = soup_uri_to_string (soup_message_get_uri (msg), FALSE);
64 redir_uri = g_strdup_printf ("%s/", uri);
65 soup_message_headers_append (msg->response_headers,
66 "Location", redir_uri);
67 soup_message_set_status (msg, SOUP_STATUS_MOVED_PERMANENTLY);
73 index_path = g_strdup_printf ("%s/index.html", path);
74 do_get (server, msg, index_path);
79 fd = open (path, O_RDONLY);
81 soup_message_set_status (msg, SOUP_STATUS_INTERNAL_SERVER_ERROR);
85 if (msg->method == SOUP_METHOD_GET) {
87 struct mapping *mapping = g_slice_new (struct mapping);
90 mapping->start = mmap (NULL, st.st_size, PROT_READ,
92 mapping->length = st.st_size;
93 buffer = soup_buffer_new_with_owner (mapping->start,
95 mapping, free_mapping);
96 soup_message_body_append_buffer (msg->response_body, buffer);
97 soup_buffer_free (buffer);
101 buf = g_malloc (st.st_size);
102 read (fd, buf, st.st_size);
104 soup_message_body_append (msg->response_body, SOUP_MEMORY_TAKE,
107 } else /* msg->method == SOUP_METHOD_HEAD */ {
110 /* We could just use the same code for both GET and
111 * HEAD. But we'll optimize and avoid the extra
114 length = g_strdup_printf ("%lu", (gulong)st.st_size);
115 soup_message_headers_append (msg->response_headers,
116 "Content-Length", length);
120 soup_message_set_status (msg, SOUP_STATUS_OK);
124 do_put (SoupServer *server, SoupMessage *msg, const char *path)
128 gboolean created = TRUE;
130 if (stat (path, &st) != -1) {
131 const char *match = soup_message_headers_get (msg->request_headers, "If-None-Match");
132 if (match && !strcmp (match, "*")) {
133 soup_message_set_status (msg, SOUP_STATUS_CONFLICT);
137 if (!S_ISREG (st.st_mode)) {
138 soup_message_set_status (msg, SOUP_STATUS_FORBIDDEN);
145 f = fopen (path, "w");
147 soup_message_set_status (msg, SOUP_STATUS_INTERNAL_SERVER_ERROR);
151 fwrite (msg->request_body->data, 1, msg->request_body->length, f);
154 soup_message_set_status (msg, created ? SOUP_STATUS_CREATED : SOUP_STATUS_OK);
158 server_callback (SoupServer *server, SoupMessage *msg,
159 const char *path, GHashTable *query,
160 SoupClientContext *context, gpointer data)
163 SoupMessageHeadersIter iter;
164 const char *name, *value;
166 printf ("%s %s HTTP/1.%d\n", msg->method, path,
167 soup_message_get_http_version (msg));
168 soup_message_headers_iter_init (&iter, msg->request_headers);
169 while (soup_message_headers_iter_next (&iter, &name, &value))
170 printf ("%s: %s\n", name, value);
171 if (msg->request_body->length)
172 printf ("%s\n", msg->request_body->data);
174 file_path = g_strdup_printf (".%s", path);
176 if (msg->method == SOUP_METHOD_GET || msg->method == SOUP_METHOD_HEAD)
177 do_get (server, msg, file_path);
178 else if (msg->method == SOUP_METHOD_PUT)
179 do_put (server, msg, file_path);
181 soup_message_set_status (msg, SOUP_STATUS_NOT_IMPLEMENTED);
184 printf (" -> %d %s\n\n", msg->status_code, msg->reason_phrase);
190 /* Exit cleanly on ^C in case we're valgrinding. */
195 main (int argc, char **argv)
198 SoupServer *server, *ssl_server;
200 int port = SOUP_ADDRESS_ANY_PORT;
201 int ssl_port = SOUP_ADDRESS_ANY_PORT;
202 const char *ssl_cert_file = NULL, *ssl_key_file = NULL;
205 g_thread_init (NULL);
206 signal (SIGINT, quit);
208 while ((opt = getopt (argc, argv, "p:k:c:s:")) != -1) {
211 port = atoi (optarg);
214 ssl_key_file = optarg;
217 ssl_cert_file = optarg;
220 ssl_port = atoi (optarg);
223 fprintf (stderr, "Usage: %s [-p port] [-c ssl-cert-file -k ssl-key-file [-s ssl-port]]\n",
229 server = soup_server_new (SOUP_SERVER_PORT, port,
230 SOUP_SERVER_SERVER_HEADER, "simple-httpd ",
233 fprintf (stderr, "Unable to bind to server port %d\n", port);
236 soup_server_add_handler (server, NULL,
237 server_callback, NULL, NULL);
238 printf ("\nStarting Server on port %d\n",
239 soup_server_get_port (server));
240 soup_server_run_async (server);
242 if (ssl_cert_file && ssl_key_file) {
243 ssl_server = soup_server_new (
244 SOUP_SERVER_PORT, ssl_port,
245 SOUP_SERVER_SSL_CERT_FILE, ssl_cert_file,
246 SOUP_SERVER_SSL_KEY_FILE, ssl_key_file,
250 fprintf (stderr, "Unable to bind to SSL server port %d\n", ssl_port);
253 soup_server_add_handler (ssl_server, NULL,
254 server_callback, NULL, NULL);
255 printf ("Starting SSL Server on port %d\n",
256 soup_server_get_port (ssl_server));
257 soup_server_run_async (ssl_server);
260 printf ("\nWaiting for requests...\n");
262 loop = g_main_loop_new (NULL, TRUE);
263 g_main_loop_run (loop);