1 /* 2 This file is part of libmicrohttpd 3 Copyright (C) 2007, 2008 Christian Grothoff (and other contributing authors) 4 5 This library is free software; you can redistribute it and/or 6 modify it under the terms of the GNU Lesser General Public 7 License as published by the Free Software Foundation; either 8 version 2.1 of the License, or (at your option) any later version. 9 10 This library is distributed in the hope that it will be useful, 11 but WITHOUT ANY WARRANTY; without even the implied warranty of 12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 13 Lesser General Public License for more details. 14 15 You should have received a copy of the GNU Lesser General Public 16 License along with this library; if not, write to the Free Software 17 Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA 18 */ 19 /** 20 * @file fileserver_example_external_select.c 21 * @brief minimal example for how to use libmicrohttpd to server files 22 * @author Christian Grothoff 23 */ 24 25 #include "platform.h" 26 #include <microhttpd.h> 27 #include <sys/stat.h> 28 #include <unistd.h> 29 30 #define PAGE "<html><head><title>File not found</title></head><body>File not found</body></html>" 31 32 static ssize_t 33 file_reader (void *cls, uint64_t pos, char *buf, size_t max) 34 { 35 FILE *file = cls; 36 37 (void) fseek (file, pos, SEEK_SET); 38 return fread (buf, 1, max, file); 39 } 40 41 static void 42 free_callback (void *cls) 43 { 44 FILE *file = cls; 45 fclose (file); 46 } 47 48 static int 49 ahc_echo (void *cls, 50 struct MHD_Connection *connection, 51 const char *url, 52 const char *method, 53 const char *version, 54 const char *upload_data, 55 size_t *upload_data_size, void **ptr) 56 { 57 static int aptr; 58 struct MHD_Response *response; 59 int ret; 60 FILE *file; 61 struct stat buf; 62 63 if (0 != strcmp (method, MHD_HTTP_METHOD_GET)) 64 return MHD_NO; /* unexpected method */ 65 if (&aptr != *ptr) 66 { 67 /* do never respond on first call */ 68 *ptr = &aptr; 69 return MHD_YES; 70 } 71 *ptr = NULL; /* reset when done */ 72 if ( (0 == stat (&url[1], &buf)) && 73 (S_ISREG (buf.st_mode)) ) 74 file = fopen (&url[1], "rb"); 75 else 76 file = NULL; 77 if (file == NULL) 78 { 79 response = MHD_create_response_from_buffer (strlen (PAGE), 80 (void *) PAGE, 81 MHD_RESPMEM_PERSISTENT); 82 ret = MHD_queue_response (connection, MHD_HTTP_NOT_FOUND, response); 83 MHD_destroy_response (response); 84 } 85 else 86 { 87 response = MHD_create_response_from_callback (buf.st_size, 32 * 1024, /* 32k page size */ 88 &file_reader, 89 file, 90 &free_callback); 91 if (response == NULL) 92 { 93 fclose (file); 94 return MHD_NO; 95 } 96 ret = MHD_queue_response (connection, MHD_HTTP_OK, response); 97 MHD_destroy_response (response); 98 } 99 return ret; 100 } 101 102 int 103 main (int argc, char *const *argv) 104 { 105 struct MHD_Daemon *d; 106 time_t end; 107 time_t t; 108 struct timeval tv; 109 fd_set rs; 110 fd_set ws; 111 fd_set es; 112 MHD_socket max; 113 MHD_UNSIGNED_LONG_LONG mhd_timeout; 114 115 if (argc != 3) 116 { 117 printf ("%s PORT SECONDS-TO-RUN\n", argv[0]); 118 return 1; 119 } 120 d = MHD_start_daemon (MHD_USE_DEBUG, 121 atoi (argv[1]), 122 NULL, NULL, &ahc_echo, PAGE, MHD_OPTION_END); 123 if (d == NULL) 124 return 1; 125 end = time (NULL) + atoi (argv[2]); 126 while ((t = time (NULL)) < end) 127 { 128 tv.tv_sec = end - t; 129 tv.tv_usec = 0; 130 max = 0; 131 FD_ZERO (&rs); 132 FD_ZERO (&ws); 133 FD_ZERO (&es); 134 if (MHD_YES != MHD_get_fdset (d, &rs, &ws, &es, &max)) 135 break; /* fatal internal error */ 136 if (MHD_get_timeout (d, &mhd_timeout) == MHD_YES) 137 { 138 if (((MHD_UNSIGNED_LONG_LONG)tv.tv_sec) < mhd_timeout / 1000LL) 139 { 140 tv.tv_sec = mhd_timeout / 1000LL; 141 tv.tv_usec = (mhd_timeout - (tv.tv_sec * 1000LL)) * 1000LL; 142 } 143 } 144 select (max + 1, &rs, &ws, &es, &tv); 145 MHD_run (d); 146 } 147 MHD_stop_daemon (d); 148 return 0; 149 } 150