/[osn-commons]/trunk/freehttpd/freehttpd.c
ViewVC logotype

Diff of /trunk/freehttpd/freehttpd.c

Parent Directory Parent Directory | Revision Log Revision Log | View Patch Patch

revision 44 by rakinar2, Sat Aug 10 18:33:37 2024 UTC revision 45 by rakinar2, Sun Aug 11 17:35:46 2024 UTC
# Line 1  Line 1 
1  #include "freehttpd.h"  #include "freehttpd.h"
2    #include "log.h"
3  #include "request.h"  #include "request.h"
4    #include "response.h"
5    
6  #include <arpa/inet.h>  #include <arpa/inet.h>
7    #include <dirent.h>
8    #include <errno.h>
9    #include <magic.h>
10    #include <stdarg.h>
11  #include <stdbool.h>  #include <stdbool.h>
12  #include <stdint.h>  #include <stdint.h>
13  #include <stdio.h>  #include <stdio.h>
14  #include <stdlib.h>  #include <stdlib.h>
15  #include <string.h>  #include <string.h>
16  #include <sys/socket.h>  #include <sys/socket.h>
17    #include <sys/stat.h>
18  #include <sys/types.h>  #include <sys/types.h>
19    #include <time.h>
20  #include <unistd.h>  #include <unistd.h>
21    
22  struct freehttpd  struct freehttpd
23  {  {
24      int sockfd;      int sockfd;
25        magic_t magic;
26      freehttpd_config_t *config;      freehttpd_config_t *config;
27  };  };
28    
# Line 30  freehttpd_config_init () Line 40  freehttpd_config_init ()
40      config->max_method_len = 16;      config->max_method_len = 16;
41      config->max_uri_len = 8192;      config->max_uri_len = 8192;
42      config->max_version_len = 16;      config->max_version_len = 16;
43        config->docroot = NULL;
44    
45      return config;      return config;
46  }  }
# Line 40  freehttpd_config_free (freehttpd_config_ Line 51  freehttpd_config_free (freehttpd_config_
51      if (config == NULL)      if (config == NULL)
52          return;          return;
53    
54        free (config->docroot);
55      free (config->addr);      free (config->addr);
56      free (config);      free (config);
57  }  }
58    
59  freehttpd_t *  freehttpd_t *
60  freehttpd_init ()  freehttpd_init (magic_t magic)
61  {  {
62      freehttpd_t *freehttpd = calloc (1, sizeof (freehttpd_t));      freehttpd_t *freehttpd = calloc (1, sizeof (freehttpd_t));
63    
# Line 54  freehttpd_init () Line 66  freehttpd_init ()
66    
67      freehttpd->sockfd = -1;      freehttpd->sockfd = -1;
68      freehttpd->config = freehttpd_config_init ();      freehttpd->config = freehttpd_config_init ();
69        freehttpd->magic = magic;
70      return freehttpd;      return freehttpd;
71  }  }
72    
73  ecode_t  ecode_t
74  freehttpd_setopt (freehttpd_t *freehttpd, freehttpd_opt_t opt, void *value)  freehttpd_setopt (freehttpd_t *freehttpd, freehttpd_opt_t opt, void *value)
75  {  {
     if (opt >= CONFIG_OPTION_COUNT)  
         return E_UNKNOWN_OPT;  
   
76      switch (opt)      switch (opt)
77          {          {
78          case FREEHTTPD_CONFIG_PORT:          case FREEHTTPD_CONFIG_PORT:
# Line 90  freehttpd_setopt (freehttpd_t *freehttpd Line 100  freehttpd_setopt (freehttpd_t *freehttpd
100              freehttpd->config->max_version_len = *(size_t *) value;              freehttpd->config->max_version_len = *(size_t *) value;
101              break;              break;
102    
103            case FREEHTTPD_CONFIG_DOCROOT:
104                freehttpd->config->docroot
105                    = value == NULL ? NULL : strdup ((const char *) value);
106    
107                if (value != NULL)
108                    freehttpd->config->_docroot_length
109                        = strlen (freehttpd->config->docroot);
110                break;
111    
112          default:          default:
113              return E_UNKNOWN_OPT;              return E_UNKNOWN_OPT;
114          }          }
# Line 160  freehttpd_listen (freehttpd_t *freehttpd Line 179  freehttpd_listen (freehttpd_t *freehttpd
179  }  }
180    
181  static ecode_t  static ecode_t
182    freehttpd_send_error (FILE *stream, int sockfd, freehttpd_status_t status)
183    {
184        freehttpd_response_t *response = freehttpd_response_init ("1.1", 3, status);
185    
186        if (response == NULL)
187            return E_LIBC_MALLOC;
188    
189        if (stream == NULL)
190            stream = fdopen (sockfd, "w");
191    
192        freehttpd_response_add_default_headers (response);
193        freehttpd_response_add_header (response, "Content-Type",
194                                       "text/html; charset=\"utf-8\"", 12, 26);
195    
196        response->body = NULL;
197        (void) (asprintf (&response->body,
198                          "<center><h1>%d %s</h1><hr><p>freehttpd</p></center>\r\n",
199                          response->status.code, response->status.text)
200                + 1);
201    
202        response->body_length = strlen (response->body);
203    
204        char *len = NULL;
205        (void) (asprintf (&len, "%lu", response->body_length) + 1);
206    
207        freehttpd_response_add_header (response, "Content-Length", len, 14,
208                                       strlen (len));
209    
210        ecode_t ret = freehttpd_response_send (response, stream);
211    
212        if (ret != E_OK)
213            log_err (LOG_ERR "failed to send error response: %i\n", ret);
214    
215        fflush (stream);
216        free (len);
217        freehttpd_response_free (response);
218        return ret;
219    }
220    
221    static long
222    fsize (FILE *file)
223    {
224        long size;
225        long pos = ftell (file);
226        fseek (file, 0, SEEK_END);
227        size = ftell (file);
228        fseek (file, pos, SEEK_SET);
229        return size;
230    }
231    
232    static void
233    iasprintf (char **strp, const char *fmt, ...)
234    {
235        va_list ap;
236        va_start (ap, fmt);
237        (void) (vasprintf (strp, fmt, ap) + 1);
238        va_end (ap);
239    }
240    
241    static ecode_t
242    freehttpd_respond_dindex (freehttpd_t *freehttpd, freehttpd_request_t *request,
243                              freehttpd_response_t *response, FILE *stream,
244                              const char *rpath)
245    {
246        freehttpd_response_set_status (response, FREEHTTPD_STATUS_OK);
247        ecode_t code = freehttpd_response_send (response, stream);
248    
249        if (code != E_OK)
250            return code;
251    
252        bool is_root = strcmp (freehttpd->config->docroot, rpath) == 0;
253        bool http1_0 = strcmp (request->version, "1.0") == 0;
254    
255        fprintf (stream, "Content-Type: text/html; charset=\"utf-8\"\r\n");
256    
257        if (http1_0)
258            fprintf (stream, "Content-Length: -1\r\n");
259        else
260            fprintf (stream, "Transfer-Encoding: chunked\r\n");
261    
262        fprintf (stream, "\r\n");
263    
264        DIR *dir = opendir (rpath);
265    
266        if (dir == NULL)
267            return E_SYSCALL_READ;
268    
269        struct dirent *entry = NULL;
270        char *out_buf = NULL;
271    
272        iasprintf (
273            &out_buf,
274            "<!DOCTYPE html PUBLIC \"-//W3C//DTD HTML 3.2//EN\">\r\n"
275            "<html>\r\n"
276            "<head>\r\n"
277            "<title>Index of %s</title>\r\n"
278            "</head>\r\n"
279            "<body>\r\n"
280            "<h1>Index of %s</h1>\r\n"
281            "<table>\r\n"
282            "<tr>\r\n"
283            "    <th>\r\n"
284            "        <img\r\n"
285            "            src=\"https://httpd.apache.org/icons/blank.gif\"\r\n"
286            "            alt=\"[ICO]\"\r\n"
287            "        />\r\n"
288            "    </th>\r\n"
289            "    <th><a href=\"?C=N;O=D\">Name</a></th>\r\n"
290            "    <th><a href=\"?C=M;O=A\">Last modified</a></th>\r\n"
291            "    <th><a href=\"?C=S;O=A\">Size</a></th>\r\n"
292            "    <th><a href=\"?C=D;O=A\">Description</a></th>\r\n"
293            "</tr>\r\n"
294            "<tr>\r\n"
295            "    <th colspan=\"5\"><hr /></th>\r\n"
296            "</tr>\r\n",
297            request->uri, request->uri);
298    
299        fprintf (stream, "%lx\r\n%s\r\n", strlen (out_buf), out_buf);
300        free (out_buf);
301    
302        if (!is_root)
303            {
304                iasprintf (&out_buf,
305                           "<tr>\r\n"
306                           "<td valign=\"top\">\r\n"
307                           "<img\r\n"
308                           "src=\"https://httpd.apache.org/icons/back.gif\"\r\n"
309                           "alt=\"[DIR]\"\r\n"
310                           "/>\r\n"
311                           "</td>\r\n"
312                           "<td><a href=\"..\">Parent Directory</a></td>\r\n"
313                           "<td>&nbsp;</td>\r\n"
314                           "<td align=\"right\"> - </td>\r\n"
315                           "<td>&nbsp;</td>\r\n"
316                           "</tr>\r\n");
317    
318                fprintf (stream, "%lx\r\n%s\r\n", strlen (out_buf), out_buf);
319                free (out_buf);
320            }
321    
322        while ((entry = readdir (dir)) != NULL)
323            {
324                if (strcmp (entry->d_name, ".") == 0
325                    || strcmp (entry->d_name, "..") == 0)
326                    continue;
327    
328                struct stat st = { 0 };
329                char path[PATH_MAX] = { 0 };
330                char mtime_str[64] = { 0 };
331    
332                strcpy (path, rpath);
333                strcat (path, "/");
334                strcat (path, entry->d_name);
335    
336                if (lstat (path, &st) < 0)
337                    continue;
338    
339                struct tm *mtime = localtime (&st.st_mtime);
340                strftime (mtime_str, sizeof (mtime_str), "%Y-%m-%d %H:%M", mtime);
341    
342                iasprintf (&out_buf,
343                           "<tr>\r\n"
344                           "<td valign=\"top\">\r\n"
345                           "<img\r\n"
346                           "src=\"https://httpd.apache.org/icons/%s.gif\"\r\n"
347                           "alt=\"[DIR]\"\r\n"
348                           "/>\r\n"
349                           "</td>\r\n",
350                           S_ISDIR (st.st_mode) ? "folder" : "unknown");
351    
352                fprintf (stream, "%lx\r\n%s\r\n", strlen (out_buf), out_buf);
353                free (out_buf);
354    
355                const char *leading_slash = S_ISDIR (st.st_mode) ? "/" : "";
356                iasprintf (&out_buf,
357                           "<td><a href=\"%s%s%s%s\">%s%s</a></td>\r\n"
358                           "<td align=\"right\">%s</td>\r\n"
359                           "<td align=\"right\"> - </td>\r\n"
360                           "</tr>\r\n",
361                           request->uri,
362                           request->uri[request->uri_length - 1] == '/' ? "" : "/",
363                           entry->d_name, leading_slash, entry->d_name,
364                           leading_slash, mtime_str);
365    
366                fprintf (stream, "%lx\r\n%s\r\n", strlen (out_buf), out_buf);
367                free (out_buf);
368            }
369    
370        iasprintf (&out_buf,
371                   "<tr>\r\n"
372                   "    <th colspan=\"5\"><hr /></th>\r\n"
373                   "</tr>\r\n"
374                   "</table><address>freehttpd/1.0.0-beta.1 (Ubuntu 24.04 "
375                   "LTS) Server at localhost Port 8080</address>\r\n"
376                   "</body>\r\n"
377                   "</html>\r\n");
378    
379        fprintf (stream, "%lx\r\n%s\r\n", strlen (out_buf), out_buf);
380        free (out_buf);
381    
382        fprintf (stream, "0\r\n\r\n");
383        closedir (dir);
384        return E_OK;
385    }
386    
387    static ecode_t
388    freehttpd_respond (freehttpd_t *freehttpd, freehttpd_request_t *request,
389                       freehttpd_response_t *response, FILE *stream)
390    {
391        freehttpd_response_add_default_headers (response);
392    
393        const char *docroot = freehttpd->config->docroot;
394        char *rpath = NULL;
395        freehttpd_status_t status = FREEHTTPD_STATUS_OK;
396    
397        if (docroot == NULL)
398            {
399                status = FREEHTTPD_STATUS_INTERNAL_SERVER_ERROR;
400                goto freehttpd_respond_error;
401            }
402    
403        char *path = request->path;
404    
405        if (path == NULL)
406            {
407                status = FREEHTTPD_STATUS_INTERNAL_SERVER_ERROR;
408                goto freehttpd_respond_error;
409            }
410    
411        char *fs_path = NULL;
412    
413        if (asprintf (&fs_path, "%s%s", docroot, path) < 0)
414            {
415                status = FREEHTTPD_STATUS_INTERNAL_SERVER_ERROR;
416                return E_LIBC_MALLOC;
417            }
418    
419        rpath = realpath (fs_path, NULL);
420    
421        if (rpath == NULL)
422            {
423                status = FREEHTTPD_STATUS_NOT_FOUND;
424                goto freehttpd_respond_error;
425            }
426    
427        if (strncmp (docroot, rpath, freehttpd->config->_docroot_length) != 0)
428            {
429                status = FREEHTTPD_STATUS_FORBIDDEN;
430                goto freehttpd_respond_error;
431            }
432    
433        log_msg (LOG_DEBUG "rpath: %s\n", rpath);
434        struct stat st = { 0 };
435    
436        if (lstat (rpath, &st) < 0)
437            {
438                status = FREEHTTPD_STATUS_INTERNAL_SERVER_ERROR;
439                goto freehttpd_respond_error;
440            }
441    
442        if (S_ISDIR (st.st_mode))
443            {
444                ecode_t code = freehttpd_respond_dindex (freehttpd, request,
445                                                         response, stream, rpath);
446    
447                free (rpath);
448                free (fs_path);
449                return code;
450            }
451    
452        FILE *file = fopen (rpath, "r");
453    
454        if (file == NULL)
455            {
456                switch (errno)
457                    {
458                    case EACCES:
459                    case EISDIR:
460                        status = FREEHTTPD_STATUS_FORBIDDEN;
461                        break;
462                    case ENOENT:
463                        status = FREEHTTPD_STATUS_NOT_FOUND;
464                        break;
465                    default:
466                        status = FREEHTTPD_STATUS_INTERNAL_SERVER_ERROR;
467                        break;
468                    }
469    
470                goto freehttpd_respond_error;
471            }
472    
473        const char *content_type = NULL;
474        size_t rpath_len = strlen (rpath);
475    
476        if (rpath_len > 4 && strcmp (rpath + rpath_len - 4, ".css") == 0)
477            content_type = "text/css";
478        else if (rpath_len > 5 && strcmp (rpath + rpath_len - 5, ".html") == 0)
479            content_type = "text/html";
480        else if (rpath_len > 4 && strcmp (rpath + rpath_len - 3, ".js") == 0)
481            content_type = "application/javascript";
482        else
483            {
484                magic_descriptor (freehttpd->magic, fileno (file));
485    
486                if (content_type == NULL)
487                    content_type = "application/octet-stream";
488            }
489    
490        fseek (file, 0, SEEK_SET);
491    
492        long file_size = fsize (file);
493        char buffer[1024] = { 0 };
494    
495        if (file_size == -1)
496            {
497                status = FREEHTTPD_STATUS_INTERNAL_SERVER_ERROR;
498                goto freehttpd_respond_error;
499            }
500    
501        freehttpd_response_set_status (response, status);
502        ecode_t code = freehttpd_response_send (response, stream);
503    
504        if (code != E_OK)
505            {
506                free (rpath);
507                free (fs_path);
508                fclose (file);
509                return code;
510            }
511    
512        char etag[128] = { 0 };
513        bool http1_0 = strcmp (request->version, "1.0") == 0;
514        snprintf (etag, sizeof (etag), "\"%lx-%lx\"", st.st_mtime, file_size);
515    
516        fprintf (stream, "Content-Type: %s\r\n", content_type);
517    
518        if (http1_0)
519            fprintf (stream, "Content-Length: %ld\r\n", file_size);
520        else
521            fprintf (stream, "Transfer-Encoding: chunked\r\n");
522    
523        fprintf (stream, "ETag: %s\r\n", etag);
524        fprintf (stream, "\r\n");
525    
526        while (file_size > 0)
527            {
528                size_t read_size = 0;
529    
530                errno = 0;
531    
532                if ((read_size = fread (buffer, 1, sizeof (buffer), file)) == 0
533                    && errno != 0)
534                    {
535                        freehttpd_response_set_status (
536                            response, FREEHTTPD_STATUS_INTERNAL_SERVER_ERROR);
537                        free (rpath);
538                        free (fs_path);
539                        fclose (file);
540                        return E_SYSCALL_READ;
541                    }
542    
543                if (!http1_0)
544                    fprintf (stream, "%lx\r\n", read_size);
545    
546                if (fwrite (buffer, 1, read_size, stream) != read_size)
547                    {
548                        free (rpath);
549                        free (fs_path);
550                        fclose (file);
551                        return E_SYSCALL_WRITE;
552                    }
553    
554                if (!http1_0)
555                    fprintf (stream, "\r\n");
556    
557                if (file_size >= (long int) read_size)
558                    file_size -= (long int) read_size;
559                else
560                    file_size = 0;
561            }
562    
563        if (!http1_0)
564            fprintf (stream, "0\r\n\r\n");
565    
566        fclose (file);
567    freehttpd_respond_error:
568        freehttpd_response_set_status (response, status);
569    
570        if (status != FREEHTTPD_STATUS_OK)
571            freehttpd_send_error (stream, fileno (stream), status);
572    
573        free (rpath);
574        free (fs_path);
575        return E_OK;
576    }
577    
578    static ecode_t
579  freehttpd_loop (freehttpd_t *freehttpd)  freehttpd_loop (freehttpd_t *freehttpd)
580  {  {
581      while (true)      while (true)
# Line 179  freehttpd_loop (freehttpd_t *freehttpd) Line 595  freehttpd_loop (freehttpd_t *freehttpd)
595    
596              if (code != E_OK)              if (code != E_OK)
597                  {                  {
598                      close (client_sockfd);                      log_err (LOG_ERR "failed to parse request: %i\n", code);
599                        freehttpd_send_error (NULL, client_sockfd,
600                                              FREEHTTPD_STATUS_BAD_REQUEST);
601                      continue;                      continue;
602                  }                  }
603    
604              fprintf (stdout, "Incoming: %s %s %s\n", request->method,              freehttpd_response_t *response = freehttpd_response_init (
605                       request->uri, request->version);                  request->version, request->version_length, FREEHTTPD_STATUS_OK);
606    
607              FILE *client = fdopen (client_sockfd, "w");              if (response == NULL)
   
             if (client == NULL)  
608                  {                  {
609                        log_err (LOG_ERR "failed to init response\n");
610                        freehttpd_send_error (
611                            NULL, client_sockfd,
612                            FREEHTTPD_STATUS_INTERNAL_SERVER_ERROR);
613                      freehttpd_request_free (request);                      freehttpd_request_free (request);
614                        close (client_sockfd);
615                      continue;                      continue;
616                  }                  }
617    
618              fprintf (client, "%s 200 OK\r\n", request->version);              FILE *stream = fdopen (client_sockfd, "w");
619              fprintf (client, "Server: FreeHTTPD\r\n");  
620              fprintf (client, "Content-Type: text/html; charset=\"utf-8\"\r\n");              code = freehttpd_respond (freehttpd, request, response, stream);
621              fprintf (client, "Content-Length: 24\r\n");  
622              fprintf (client, "Connection: close\r\n");              log_msg (LOG_INFO "%s %s HTTP/%s - %d %s\n", request->method,
623              fprintf (client, "\r\n");                       request->uri, request->version, response->status.code,
624              fprintf (client, "<h1>Hello, World!</h1>\r\n");                       response->status.text);
625              fflush (client);  
626                if (code != E_OK)
627                    log_err (LOG_ERR "failed to send response: %i\n", code);
628    
629                fclose (stream);
630                freehttpd_response_free (response);
631              freehttpd_request_free (request);              freehttpd_request_free (request);
             fclose (client);  
632          }          }
633    
634      return E_OK;      return E_OK;

Legend:
Removed from v.44  
changed lines
  Added in v.45

[email protected]
ViewVC Help
Powered by ViewVC 1.1.26