Compare commits
	
		
			21 Commits
		
	
	
		
			v4.5
			...
			f0b27b3b37
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| f0b27b3b37 | |||
| ee8aedce91 | |||
| 0648c75baa | |||
| b6ba58d406 | |||
| abe0e326cb | |||
| 4fe067ed7d | |||
| 41e12d6293 | |||
| a738f1abfe | |||
| b6309eec39 | |||
| 9923a76ba7 | |||
| ea4cdff233 | |||
| 6526b5cbcb | |||
| 0119945e03 | |||
| 3fe1fe023a | |||
| bc1c6d3498 | |||
| 50bb537074 | |||
| c060ee5bb6 | |||
| 4062883cb3 | |||
| e0e44e9c26 | |||
| f0d8a3db4c | |||
| 557e176d3d | 
							
								
								
									
										6
									
								
								Makefile
									
									
									
									
									
								
							
							
						
						
									
										6
									
								
								Makefile
									
									
									
									
									
								
							| @@ -1,6 +1,6 @@ | |||||||
|  |  | ||||||
| CC=gcc | CC=gcc | ||||||
| CFLAGS=-std=gnu11 -Wall | CFLAGS=-std=gnu11 -Wall -Wno-unused-but-set-variable | ||||||
| LIBS=-lssl -lcrypto -lmagic -lz -lmaxminddb -lbrotlienc | LIBS=-lssl -lcrypto -lmagic -lz -lmaxminddb -lbrotlienc | ||||||
|  |  | ||||||
| DEBIAN_OPTS=-D CACHE_MAGIC_FILE="\"/usr/share/file/magic.mgc\"" -D PHP_FPM_SOCKET="\"/var/run/php/php7.4-fpm.sock\"" | DEBIAN_OPTS=-D CACHE_MAGIC_FILE="\"/usr/share/file/magic.mgc\"" -D PHP_FPM_SOCKET="\"/var/run/php/php7.4-fpm.sock\"" | ||||||
| @@ -16,11 +16,11 @@ permit: | |||||||
| compile: | compile: | ||||||
| 	@mkdir -p bin | 	@mkdir -p bin | ||||||
| 	$(CC) src/lib/*.c -o bin/libnecrondaserver.so --shared -fPIC $(CFLAGS) $(LIBS) | 	$(CC) src/lib/*.c -o bin/libnecrondaserver.so --shared -fPIC $(CFLAGS) $(LIBS) | ||||||
| 	$(CC) src/necronda-server.c -o bin/necronda-server $(CFLAGS) $(LIBS) \ | 	$(CC) src/server.c src/client.c -o bin/necronda-server $(CFLAGS) $(LIBS) \ | ||||||
| 		-Lbin -lnecrondaserver -Wl,-rpath=$(shell pwd)/bin | 		-Lbin -lnecrondaserver -Wl,-rpath=$(shell pwd)/bin | ||||||
|  |  | ||||||
| compile-prod: | compile-prod: | ||||||
| 	@mkdir -p bin | 	@mkdir -p bin | ||||||
| 	$(CC) src/lib/*.c -o bin/libnecrondaserver.so --shared -fPIC $(CFLAGS) $(LIBS) $(DEBIAN_OPTS) -O3 | 	$(CC) src/lib/*.c -o bin/libnecrondaserver.so --shared -fPIC $(CFLAGS) $(LIBS) $(DEBIAN_OPTS) -O3 | ||||||
| 	$(CC) src/necronda-server.c -o bin/necronda-server $(CFLAGS) $(LIBS) $(DEBIAN_OPTS) -O3 \ | 	$(CC) src/server.c src/client.c -o bin/necronda-server $(CFLAGS) $(LIBS) $(DEBIAN_OPTS) -O3 \ | ||||||
| 		-Lbin -lnecrondaserver -Wl,-rpath=$(shell pwd)/bin | 		-Lbin -lnecrondaserver -Wl,-rpath=$(shell pwd)/bin | ||||||
|   | |||||||
| @@ -7,9 +7,12 @@ Necronda web server | |||||||
| * Full IPv4 and IPv6 support | * Full IPv4 and IPv6 support | ||||||
| * TLS Server Name Inspection (SNI) | * TLS Server Name Inspection (SNI) | ||||||
| * Serving local files via HTTP and HTTPS | * Serving local files via HTTP and HTTPS | ||||||
|   * File compression ([gzip](https://www.gzip.org/), [Brotli](https://www.brotli.org/)) and disk cache for compressed files |   * File compression ([gzip](https://www.gzip.org/), [Brotli](https://www.brotli.org/)) | ||||||
|  |   * Disk cache for compressed files | ||||||
| * Reverse proxy for other HTTP and HTTPS servers | * Reverse proxy for other HTTP and HTTPS servers | ||||||
|  |   * Transparent WebSocket reverse proxy **[WIP]** | ||||||
| * FastCGI support (e.g. [PHP-FPM](https://php-fpm.org/)) | * FastCGI support (e.g. [PHP-FPM](https://php-fpm.org/)) | ||||||
|  |   * Automatic path info detection (e.g. `/my/file/extra/path` -> script: `/my/file.php`, path info: `extra/path`) | ||||||
| * Support for [MaxMind's GeoIP Database](https://www.maxmind.com/en/geoip2-services-and-databases) | * Support for [MaxMind's GeoIP Database](https://www.maxmind.com/en/geoip2-services-and-databases) | ||||||
| * Optional DNS reverse lookup for connecting hosts | * Optional DNS reverse lookup for connecting hosts | ||||||
| * Automatic URL rewrite (e.g. `/index.html` -> `/`, `/test.php` -> `/test`) | * Automatic URL rewrite (e.g. `/index.html` -> `/`, `/test.php` -> `/test`) | ||||||
| @@ -29,7 +32,7 @@ See [docs/example.conf](docs/example.conf) for more details. | |||||||
|  |  | ||||||
| ### Configuration | ### Configuration | ||||||
|  |  | ||||||
| * `[cert <cert-name]` - begins section for a certificate | * `[cert <cert-name>]` - begins section for a certificate | ||||||
|   * `certificate` - path to SSL certificate (or certificate chain) |   * `certificate` - path to SSL certificate (or certificate chain) | ||||||
|   * `private_key` - path to SSL private key |   * `private_key` - path to SSL private key | ||||||
| * `[host <host>]` - begins section for the virtual host `<host>` | * `[host <host>]` - begins section for the virtual host `<host>` | ||||||
|   | |||||||
| @@ -1,19 +1,24 @@ | |||||||
|  |  | ||||||
| certificate /var/cert/cert.pem |  | ||||||
| private_key /var/cert/cert.key |  | ||||||
| #geoip_dir  /var/dir | #geoip_dir  /var/dir | ||||||
| #dns_server 192.168.0.1 | #dns_server 192.168.0.1 | ||||||
|  |  | ||||||
| [localhost] | [cert cert1] | ||||||
|  | certificate /var/cert/cert.pem | ||||||
|  | private_key /var/cert/cert.key | ||||||
|  |  | ||||||
|  | [host localhost] | ||||||
| webroot     /var/www/localhost | webroot     /var/www/localhost | ||||||
| dir_mode    forbidden | dir_mode    forbidden | ||||||
|  | cert        cert1 | ||||||
|  |  | ||||||
| [me.local] | [host me.local] | ||||||
| hostname    www.example.com | hostname    www.example.com | ||||||
| port        80 | port        80 | ||||||
|  | cert        cert1 | ||||||
| http | http | ||||||
|  |  | ||||||
| [secure.local] | [host secure.local] | ||||||
| hostname    www.example.com | hostname    www.example.com | ||||||
| port        443 | port        443 | ||||||
|  | cert        cert1 | ||||||
| https | https | ||||||
|   | |||||||
							
								
								
									
										80
									
								
								src/client.c
									
									
									
									
									
								
							
							
						
						
									
										80
									
								
								src/client.c
									
									
									
									
									
								
							| @@ -5,6 +5,10 @@ | |||||||
|  * Lorenz Stechauner, 2020-12-03 |  * Lorenz Stechauner, 2020-12-03 | ||||||
|  */ |  */ | ||||||
|  |  | ||||||
|  | #include "client.h" | ||||||
|  | #include "necronda.h" | ||||||
|  | #include "server.h" | ||||||
|  |  | ||||||
| #include "lib/utils.h" | #include "lib/utils.h" | ||||||
| #include "lib/config.h" | #include "lib/config.h" | ||||||
| #include "lib/sock.h" | #include "lib/sock.h" | ||||||
| @@ -63,7 +67,8 @@ int client_request_handler(sock *client, unsigned long client_num, unsigned int | |||||||
|     char msg_buf[8192], msg_pre_buf_1[4096], msg_pre_buf_2[4096], err_msg[256]; |     char msg_buf[8192], msg_pre_buf_1[4096], msg_pre_buf_2[4096], err_msg[256]; | ||||||
|     char msg_content[1024]; |     char msg_content[1024]; | ||||||
|     char buffer[CHUNK_SIZE]; |     char buffer[CHUNK_SIZE]; | ||||||
|     char host[256], *host_ptr, *hdr_connection; |     char host[256]; | ||||||
|  |     const char *host_ptr, *hdr_connection; | ||||||
|  |  | ||||||
|     msg_buf[0] = 0; |     msg_buf[0] = 0; | ||||||
|     err_msg[0] = 0; |     err_msg[0] = 0; | ||||||
| @@ -81,7 +86,7 @@ int client_request_handler(sock *client, unsigned long client_num, unsigned int | |||||||
|     fastcgi_conn fcgi_conn = {.socket = 0, .req_id = 0}; |     fastcgi_conn fcgi_conn = {.socket = 0, .req_id = 0}; | ||||||
|     http_status custom_status; |     http_status custom_status; | ||||||
|  |  | ||||||
|     http_res res = {.version = "1.1", .status = http_get_status(501), .hdr.field_num = 0}; |     http_res res = {.version = "1.1", .status = http_get_status(501), .hdr.field_num = 0, .hdr.last_field_num = -1}; | ||||||
|     http_status_ctx ctx = {.status = 0, .origin = NONE}; |     http_status_ctx ctx = {.status = 0, .origin = NONE}; | ||||||
|  |  | ||||||
|     clock_gettime(CLOCK_MONOTONIC, &begin); |     clock_gettime(CLOCK_MONOTONIC, &begin); | ||||||
| @@ -95,9 +100,8 @@ int client_request_handler(sock *client, unsigned long client_num, unsigned int | |||||||
|     http_add_header_field(&res.hdr, "Date", http_get_date(buf0, sizeof(buf0))); |     http_add_header_field(&res.hdr, "Date", http_get_date(buf0, sizeof(buf0))); | ||||||
|     http_add_header_field(&res.hdr, "Server", SERVER_STR); |     http_add_header_field(&res.hdr, "Server", SERVER_STR); | ||||||
|     if (ret <= 0) { |     if (ret <= 0) { | ||||||
|         if (errno != 0) { |         if (errno != 0) return 1; | ||||||
|             return 1; |  | ||||||
|         } |  | ||||||
|         client_keep_alive = 0; |         client_keep_alive = 0; | ||||||
|         res.status = http_get_status(408); |         res.status = http_get_status(408); | ||||||
|         goto respond; |         goto respond; | ||||||
| @@ -126,8 +130,7 @@ int client_request_handler(sock *client, unsigned long client_num, unsigned int | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     hdr_connection = http_get_header_field(&req.hdr, "Connection"); |     hdr_connection = http_get_header_field(&req.hdr, "Connection"); | ||||||
|     client_keep_alive = hdr_connection != NULL && |     client_keep_alive = (hdr_connection != NULL && (strcmp(hdr_connection, "keep-alive") == 0 || strcmp(hdr_connection, "Keep-Alive") == 0)); | ||||||
|                         (strcmp(hdr_connection, "keep-alive") == 0 || strcmp(hdr_connection, "Keep-Alive") == 0); |  | ||||||
|     host_ptr = http_get_header_field(&req.hdr, "Host"); |     host_ptr = http_get_header_field(&req.hdr, "Host"); | ||||||
|     if (host_ptr != NULL && strlen(host_ptr) > 255) { |     if (host_ptr != NULL && strlen(host_ptr) > 255) { | ||||||
|         host[0] = 0; |         host[0] = 0; | ||||||
| @@ -153,7 +156,7 @@ int client_request_handler(sock *client, unsigned long client_num, unsigned int | |||||||
|  |  | ||||||
|     conf = get_host_config(host); |     conf = get_host_config(host); | ||||||
|     if (conf == NULL) { |     if (conf == NULL) { | ||||||
|         print("Host unknown, redirecting to default"); |         print("Unknown host, redirecting to default"); | ||||||
|         res.status = http_get_status(307); |         res.status = http_get_status(307); | ||||||
|         sprintf(buf0, "https://%s%s", DEFAULT_HOST, req.uri); |         sprintf(buf0, "https://%s%s", DEFAULT_HOST, req.uri); | ||||||
|         http_add_header_field(&res.hdr, "Location", buf0); |         http_add_header_field(&res.hdr, "Location", buf0); | ||||||
| @@ -161,7 +164,7 @@ int client_request_handler(sock *client, unsigned long client_num, unsigned int | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     http_uri uri; |     http_uri uri; | ||||||
|     unsigned char dir_mode = conf->type == CONFIG_TYPE_LOCAL ? conf->local.dir_mode : URI_DIR_MODE_NO_VALIDATION; |     unsigned char dir_mode = (conf->type == CONFIG_TYPE_LOCAL ? conf->local.dir_mode : URI_DIR_MODE_NO_VALIDATION); | ||||||
|     ret = uri_init(&uri, conf->local.webroot, req.uri, dir_mode); |     ret = uri_init(&uri, conf->local.webroot, req.uri, dir_mode); | ||||||
|     if (ret != 0) { |     if (ret != 0) { | ||||||
|         if (ret == 1) { |         if (ret == 1) { | ||||||
| @@ -218,11 +221,10 @@ int client_request_handler(sock *client, unsigned long client_num, unsigned int | |||||||
|             res.status = http_get_status(200); |             res.status = http_get_status(200); | ||||||
|             http_add_header_field(&res.hdr, "Content-Type", "message/http"); |             http_add_header_field(&res.hdr, "Content-Type", "message/http"); | ||||||
|  |  | ||||||
|             content_length = snprintf(msg_buf, sizeof(msg_buf) - content_length, "%s %s HTTP/%s\r\n", |             content_length = snprintf(msg_buf, sizeof(msg_buf) - content_length, "%s %s HTTP/%s\r\n", req.method, req.uri, req.version); | ||||||
|                                       req.method, req.uri, req.version); |  | ||||||
|             for (int i = 0; i < req.hdr.field_num; i++) { |             for (int i = 0; i < req.hdr.field_num; i++) { | ||||||
|                 content_length += snprintf(msg_buf + content_length, sizeof(msg_buf) - content_length, "%s: %s\r\n", |                 const http_field *f = &req.hdr.fields[i]; | ||||||
|                                            req.hdr.fields[i][0], req.hdr.fields[i][1]); |                 content_length += snprintf(msg_buf + content_length, sizeof(msg_buf) - content_length, "%s: %s\r\n", http_field_get_name(f), http_field_get_value(f)); | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             goto respond; |             goto respond; | ||||||
| @@ -273,13 +275,13 @@ int client_request_handler(sock *client, unsigned long client_num, unsigned int | |||||||
|                 sprintf(err_msg, "Unable to communicate with internal file cache."); |                 sprintf(err_msg, "Unable to communicate with internal file cache."); | ||||||
|                 goto respond; |                 goto respond; | ||||||
|             } |             } | ||||||
|             char *last_modified = http_format_date(uri.meta->stat.st_mtime, buf0, sizeof(buf0)); |             const char *last_modified = http_format_date(uri.meta->stat.st_mtime, buf0, sizeof(buf0)); | ||||||
|             http_add_header_field(&res.hdr, "Last-Modified", last_modified); |             http_add_header_field(&res.hdr, "Last-Modified", last_modified); | ||||||
|             sprintf(buf1, "%s; charset=%s", uri.meta->type, uri.meta->charset); |             sprintf(buf1, "%s; charset=%s", uri.meta->type, uri.meta->charset); | ||||||
|             http_add_header_field(&res.hdr, "Content-Type", buf1); |             http_add_header_field(&res.hdr, "Content-Type", buf1); | ||||||
|  |  | ||||||
|  |  | ||||||
|             char *accept_encoding = http_get_header_field(&req.hdr, "Accept-Encoding"); |             const char *accept_encoding = http_get_header_field(&req.hdr, "Accept-Encoding"); | ||||||
|             int enc = 0; |             int enc = 0; | ||||||
|             if (accept_encoding != NULL) { |             if (accept_encoding != NULL) { | ||||||
|                 if (uri.meta->filename_comp_br[0] != 0 && strstr(accept_encoding, "br") != NULL) { |                 if (uri.meta->filename_comp_br[0] != 0 && strstr(accept_encoding, "br") != NULL) { | ||||||
| @@ -306,8 +308,7 @@ int client_request_handler(sock *client, unsigned long client_num, unsigned int | |||||||
|  |  | ||||||
|             if (uri.meta->etag[0] != 0) { |             if (uri.meta->etag[0] != 0) { | ||||||
|                 if (enc) { |                 if (enc) { | ||||||
|                     sprintf(buf0, "%s-%s", uri.meta->etag, |                     sprintf(buf0, "%s-%s", uri.meta->etag, (enc & COMPRESS_BR) ? "br" : (enc & COMPRESS_GZ) ? "gzip" : ""); | ||||||
|                             (enc & COMPRESS_BR) ? "br" : (enc & COMPRESS_GZ) ? "gzip" : ""); |  | ||||||
|                     http_add_header_field(&res.hdr, "ETag", buf0); |                     http_add_header_field(&res.hdr, "ETag", buf0); | ||||||
|                 } else { |                 } else { | ||||||
|                     http_add_header_field(&res.hdr, "ETag", uri.meta->etag); |                     http_add_header_field(&res.hdr, "ETag", uri.meta->etag); | ||||||
| @@ -320,16 +321,16 @@ int client_request_handler(sock *client, unsigned long client_num, unsigned int | |||||||
|                 http_add_header_field(&res.hdr, "Cache-Control", "public, max-age=86400"); |                 http_add_header_field(&res.hdr, "Cache-Control", "public, max-age=86400"); | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             char *if_modified_since = http_get_header_field(&req.hdr, "If-Modified-Since"); |             const char *if_modified_since = http_get_header_field(&req.hdr, "If-Modified-Since"); | ||||||
|             char *if_none_match = http_get_header_field(&req.hdr, "If-None-Match"); |             const char *if_none_match = http_get_header_field(&req.hdr, "If-None-Match"); | ||||||
|             if ((if_none_match != NULL && strstr(if_none_match, uri.meta->etag) == NULL) || |             if ((if_none_match != NULL && strstr(if_none_match, uri.meta->etag) == NULL) || | ||||||
|                     (accept_if_modified_since && if_modified_since != NULL && |                 (accept_if_modified_since && if_modified_since != NULL && strcmp(if_modified_since, last_modified) == 0)) | ||||||
|                     strcmp(if_modified_since, last_modified) == 0)) { |             { | ||||||
|                 res.status = http_get_status(304); |                 res.status = http_get_status(304); | ||||||
|                 goto respond; |                 goto respond; | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             char *range = http_get_header_field(&req.hdr, "Range"); |             const char *range = http_get_header_field(&req.hdr, "Range"); | ||||||
|             if (range != NULL) { |             if (range != NULL) { | ||||||
|                 if (strlen(range) <= 6 || strncmp(range, "bytes=", 6) != 0) { |                 if (strlen(range) <= 6 || strncmp(range, "bytes=", 6) != 0) { | ||||||
|                     res.status = http_get_status(416); |                     res.status = http_get_status(416); | ||||||
| @@ -404,7 +405,7 @@ int client_request_handler(sock *client, unsigned long client_num, unsigned int | |||||||
|                 goto respond; |                 goto respond; | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             char *client_content_length = http_get_header_field(&req.hdr, "Content-Length"); |             const char *client_content_length = http_get_header_field(&req.hdr, "Content-Length"); | ||||||
|             if (client_content_length != NULL) { |             if (client_content_length != NULL) { | ||||||
|                 unsigned long client_content_len = strtoul(client_content_length, NULL, 10); |                 unsigned long client_content_len = strtoul(client_content_length, NULL, 10); | ||||||
|                 ret = fastcgi_receive(&fcgi_conn, client, client_content_len); |                 ret = fastcgi_receive(&fcgi_conn, client, client_content_len); | ||||||
| @@ -426,7 +427,7 @@ int client_request_handler(sock *client, unsigned long client_num, unsigned int | |||||||
|                 goto respond; |                 goto respond; | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             char *status = http_get_header_field(&res.hdr, "Status"); |             const char *status = http_get_header_field(&res.hdr, "Status"); | ||||||
|             if (status != NULL) { |             if (status != NULL) { | ||||||
|                 int status_code = (int) strtoul(status, NULL, 10); |                 int status_code = (int) strtoul(status, NULL, 10); | ||||||
|                 res.status = http_get_status(status_code); |                 res.status = http_get_status(status_code); | ||||||
| @@ -497,9 +498,7 @@ int client_request_handler(sock *client, unsigned long client_num, unsigned int | |||||||
|             const char *content_type = http_get_header_field(&res.hdr, "Content-Type"); |             const char *content_type = http_get_header_field(&res.hdr, "Content-Type"); | ||||||
|             const char *content_length_f = http_get_header_field(&res.hdr, "Content-Length"); |             const char *content_length_f = http_get_header_field(&res.hdr, "Content-Length"); | ||||||
|             const char *content_encoding = http_get_header_field(&res.hdr, "Content-Encoding"); |             const char *content_encoding = http_get_header_field(&res.hdr, "Content-Encoding"); | ||||||
|             if (content_encoding == NULL && content_type != NULL && content_length_f != NULL && |             if (content_encoding == NULL && content_type != NULL && content_length_f != NULL && strncmp(content_type, "text/html", 9) == 0) { | ||||||
|                 strncmp(content_type, "text/html", 9) == 0) |  | ||||||
|             { |  | ||||||
|                 long content_len = strtol(content_length_f, NULL, 10); |                 long content_len = strtol(content_length_f, NULL, 10); | ||||||
|                 if (content_len <= sizeof(msg_content) - 1) { |                 if (content_len <= sizeof(msg_content) - 1) { | ||||||
|                     ctx.status = res.status->code; |                     ctx.status = res.status->code; | ||||||
| @@ -561,13 +560,17 @@ int client_request_handler(sock *client, unsigned long client_num, unsigned int | |||||||
|             const http_doc_info *info = http_get_status_info(res.status); |             const http_doc_info *info = http_get_status_info(res.status); | ||||||
|             const http_status_msg *http_msg = http_get_error_msg(res.status); |             const http_status_msg *http_msg = http_get_error_msg(res.status); | ||||||
|  |  | ||||||
|             if (res.status->code >= 300 && res.status->code < 400 && msg_content[0] == 0) { |             if (msg_content[0] == 0) { | ||||||
|  |                 if (res.status->code >= 300 && res.status->code < 400) { | ||||||
|                     const char *location = http_get_header_field(&res.hdr, "Location"); |                     const char *location = http_get_header_field(&res.hdr, "Location"); | ||||||
|                     if (location != NULL) { |                     if (location != NULL) { | ||||||
|                     snprintf(msg_content, sizeof(msg_content), |                         snprintf(msg_content, sizeof(msg_content), "<ul>\n\t<li><a href=\"%1$s\">%1$s</a></li>\n</ul>\n", location); | ||||||
|                              "<ul>\n\t<li><a href=\"%1$s\">%1$s</a></li>\n</ul>\n", location); |  | ||||||
|                     } |                     } | ||||||
|                 } |                 } | ||||||
|  |             } else if (strncmp(msg_content, "<!DOCTYPE html>", 15) == 0 || strncmp(msg_content, "<html", 5) == 0) { | ||||||
|  |                 msg_content[0] = 0; | ||||||
|  |                 // TODO let relevant information pass? | ||||||
|  |             } | ||||||
|  |  | ||||||
|             char *rev_proxy_doc = ""; |             char *rev_proxy_doc = ""; | ||||||
|             if (conf != NULL && conf->type == CONFIG_TYPE_REVERSE_PROXY) { |             if (conf != NULL && conf->type == CONFIG_TYPE_REVERSE_PROXY) { | ||||||
| @@ -592,8 +595,7 @@ int client_request_handler(sock *client, unsigned long client_num, unsigned int | |||||||
|                 rev_proxy_doc = msg_pre_buf_2; |                 rev_proxy_doc = msg_pre_buf_2; | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             sprintf(msg_pre_buf_1, info->doc, res.status->code, res.status->msg, |             sprintf(msg_pre_buf_1, info->doc, res.status->code, res.status->msg, http_msg != NULL ? http_msg->msg : "", err_msg[0] != 0 ? err_msg : ""); | ||||||
|                     http_msg != NULL ? http_msg->msg : "", err_msg[0] != 0 ? err_msg : ""); |  | ||||||
|             content_length = snprintf(msg_buf, sizeof(msg_buf), http_default_document, res.status->code, |             content_length = snprintf(msg_buf, sizeof(msg_buf), http_default_document, res.status->code, | ||||||
|                                       res.status->msg, msg_pre_buf_1, info->mode, info->icon, info->color, host, |                                       res.status->msg, msg_pre_buf_1, info->mode, info->icon, info->color, host, | ||||||
|                                       rev_proxy_doc, msg_content[0] != 0 ? msg_content : ""); |                                       rev_proxy_doc, msg_content[0] != 0 ? msg_content : ""); | ||||||
| @@ -607,8 +609,8 @@ int client_request_handler(sock *client, unsigned long client_num, unsigned int | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     char *conn = http_get_header_field(&res.hdr, "Connection"); |     const char *conn = http_get_header_field(&res.hdr, "Connection"); | ||||||
|     int close_proxy = conn == NULL || (strcmp(conn, "keep-alive") != 0 && strcmp(conn, "Keep-Alive") != 0); |     int close_proxy = (conn == NULL || (strcmp(conn, "keep-alive") != 0 && strcmp(conn, "Keep-Alive") != 0)); | ||||||
|     http_remove_header_field(&res.hdr, "Connection", HTTP_REMOVE_ALL); |     http_remove_header_field(&res.hdr, "Connection", HTTP_REMOVE_ALL); | ||||||
|     http_remove_header_field(&res.hdr, "Keep-Alive", HTTP_REMOVE_ALL); |     http_remove_header_field(&res.hdr, "Keep-Alive", HTTP_REMOVE_ALL); | ||||||
|     if (server_keep_alive && client_keep_alive) { |     if (server_keep_alive && client_keep_alive) { | ||||||
| @@ -621,7 +623,7 @@ int client_request_handler(sock *client, unsigned long client_num, unsigned int | |||||||
|  |  | ||||||
|     http_send_response(client, &res); |     http_send_response(client, &res); | ||||||
|     clock_gettime(CLOCK_MONOTONIC, &end); |     clock_gettime(CLOCK_MONOTONIC, &end); | ||||||
|     char *location = http_get_header_field(&res.hdr, "Location"); |     const char *location = http_get_header_field(&res.hdr, "Location"); | ||||||
|     unsigned long micros = (end.tv_nsec - begin.tv_nsec) / 1000 + (end.tv_sec - begin.tv_sec) * 1000000; |     unsigned long micros = (end.tv_nsec - begin.tv_nsec) / 1000 + (end.tv_sec - begin.tv_sec) * 1000000; | ||||||
|     print("%s%s%03i %s%s%s (%s)%s", http_get_status_color(res.status), use_rev_proxy ? "-> " : "", res.status->code, |     print("%s%s%03i %s%s%s (%s)%s", http_get_status_color(res.status), use_rev_proxy ? "-> " : "", res.status->code, | ||||||
|           res.status->msg, location != NULL ? " -> " : "", location != NULL ? location : "", |           res.status->msg, location != NULL ? " -> " : "", location != NULL ? location : "", | ||||||
| @@ -652,16 +654,16 @@ int client_request_handler(sock *client, unsigned long client_num, unsigned int | |||||||
|                 snd_len += ret; |                 snd_len += ret; | ||||||
|             } |             } | ||||||
|         } else if (use_fastcgi) { |         } else if (use_fastcgi) { | ||||||
|             char *transfer_encoding = http_get_header_field(&res.hdr, "Transfer-Encoding"); |             const char *transfer_encoding = http_get_header_field(&res.hdr, "Transfer-Encoding"); | ||||||
|             int chunked = transfer_encoding != NULL && strcmp(transfer_encoding, "chunked") == 0; |             int chunked = (transfer_encoding != NULL && strcmp(transfer_encoding, "chunked") == 0); | ||||||
|  |  | ||||||
|             int flags = (chunked ? FASTCGI_CHUNKED : 0) | (use_fastcgi & (FASTCGI_COMPRESS | FASTCGI_COMPRESS_HOLD)); |             int flags = (chunked ? FASTCGI_CHUNKED : 0) | (use_fastcgi & (FASTCGI_COMPRESS | FASTCGI_COMPRESS_HOLD)); | ||||||
|             ret = fastcgi_send(&fcgi_conn, client, flags); |             ret = fastcgi_send(&fcgi_conn, client, flags); | ||||||
|         } else if (use_rev_proxy) { |         } else if (use_rev_proxy) { | ||||||
|             char *transfer_encoding = http_get_header_field(&res.hdr, "Transfer-Encoding"); |             const char *transfer_encoding = http_get_header_field(&res.hdr, "Transfer-Encoding"); | ||||||
|             int chunked = transfer_encoding != NULL && strstr(transfer_encoding, "chunked") != NULL; |             int chunked = transfer_encoding != NULL && strstr(transfer_encoding, "chunked") != NULL; | ||||||
|  |  | ||||||
|             char *content_len = http_get_header_field(&res.hdr, "Content-Length"); |             const char *content_len = http_get_header_field(&res.hdr, "Content-Length"); | ||||||
|             unsigned long len_to_send = 0; |             unsigned long len_to_send = 0; | ||||||
|             if (content_len != NULL) { |             if (content_len != NULL) { | ||||||
|                 len_to_send = strtol(content_len, NULL, 10); |                 len_to_send = strtol(content_len, NULL, 10); | ||||||
|   | |||||||
							
								
								
									
										20
									
								
								src/client.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								src/client.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,20 @@ | |||||||
|  | /** | ||||||
|  |  * Necronda Web Server | ||||||
|  |  * Client connection and request handlers (header file) | ||||||
|  |  * src/client.h | ||||||
|  |  * Lorenz Stechauner, 2022-08-16 | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | #ifndef NECRONDA_SERVER_NECRONDA_CLIENT_H | ||||||
|  | #define NECRONDA_SERVER_NECRONDA_CLIENT_H | ||||||
|  |  | ||||||
|  | #include "lib/config.h" | ||||||
|  | #include "lib/sock.h" | ||||||
|  |  | ||||||
|  | #include <arpa/inet.h> | ||||||
|  |  | ||||||
|  | host_config *get_host_config(const char *host); | ||||||
|  |  | ||||||
|  | int client_handler(sock *client, unsigned long client_num, struct sockaddr_in6 *client_addr); | ||||||
|  |  | ||||||
|  | #endif //NECRONDA_SERVER_NECRONDA_CLIENT_H | ||||||
| @@ -8,6 +8,7 @@ | |||||||
| #include "cache.h" | #include "cache.h" | ||||||
| #include "utils.h" | #include "utils.h" | ||||||
| #include "compress.h" | #include "compress.h" | ||||||
|  |  | ||||||
| #include <stdio.h> | #include <stdio.h> | ||||||
| #include <magic.h> | #include <magic.h> | ||||||
| #include <sys/ipc.h> | #include <sys/ipc.h> | ||||||
| @@ -45,14 +46,14 @@ int cache_process() { | |||||||
|  |  | ||||||
|     int shm_id = shmget(CACHE_SHM_KEY, CACHE_ENTRIES * sizeof(cache_entry), 0); |     int shm_id = shmget(CACHE_SHM_KEY, CACHE_ENTRIES * sizeof(cache_entry), 0); | ||||||
|     if (shm_id < 0) { |     if (shm_id < 0) { | ||||||
|         fprintf(stderr, ERR_STR "Unable to create shared memory: %s" CLR_STR "\n", strerror(errno)); |         fprintf(stderr, ERR_STR "Unable to create cache shared memory: %s" CLR_STR "\n", strerror(errno)); | ||||||
|         return -1; |         return -1; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     shmdt(cache); |     shmdt(cache); | ||||||
|     void *shm_rw = shmat(shm_id, NULL, 0); |     void *shm_rw = shmat(shm_id, NULL, 0); | ||||||
|     if (shm_rw == (void *) -1) { |     if (shm_rw == (void *) -1) { | ||||||
|         fprintf(stderr, ERR_STR "Unable to attach shared memory (rw): %s" CLR_STR "\n", strerror(errno)); |         fprintf(stderr, ERR_STR "Unable to attach cache shared memory (rw): %s" CLR_STR "\n", strerror(errno)); | ||||||
|         return -2; |         return -2; | ||||||
|     } |     } | ||||||
|     cache = shm_rw; |     cache = shm_rw; | ||||||
| @@ -226,20 +227,20 @@ int cache_init() { | |||||||
|  |  | ||||||
|     int shm_id = shmget(CACHE_SHM_KEY, CACHE_ENTRIES * sizeof(cache_entry), IPC_CREAT | IPC_EXCL | 0600); |     int shm_id = shmget(CACHE_SHM_KEY, CACHE_ENTRIES * sizeof(cache_entry), IPC_CREAT | IPC_EXCL | 0600); | ||||||
|     if (shm_id < 0) { |     if (shm_id < 0) { | ||||||
|         fprintf(stderr, ERR_STR "Unable to create shared memory: %s" CLR_STR "\n", strerror(errno)); |         fprintf(stderr, ERR_STR "Unable to create cache shared memory: %s" CLR_STR "\n", strerror(errno)); | ||||||
|         return -2; |         return -2; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     void *shm = shmat(shm_id, NULL, SHM_RDONLY); |     void *shm = shmat(shm_id, NULL, SHM_RDONLY); | ||||||
|     if (shm == (void *) -1) { |     if (shm == (void *) -1) { | ||||||
|         fprintf(stderr, ERR_STR "Unable to attach shared memory (ro): %s" CLR_STR "\n", strerror(errno)); |         fprintf(stderr, ERR_STR "Unable to attach cache shared memory (ro): %s" CLR_STR "\n", strerror(errno)); | ||||||
|         return -3; |         return -3; | ||||||
|     } |     } | ||||||
|     cache = shm; |     cache = shm; | ||||||
|  |  | ||||||
|     void *shm_rw = shmat(shm_id, NULL, 0); |     void *shm_rw = shmat(shm_id, NULL, 0); | ||||||
|     if (shm_rw == (void *) -1) { |     if (shm_rw == (void *) -1) { | ||||||
|         fprintf(stderr, ERR_STR "Unable to attach shared memory (rw): %s" CLR_STR "\n", strerror(errno)); |         fprintf(stderr, ERR_STR "Unable to attach cache shared memory (rw): %s" CLR_STR "\n", strerror(errno)); | ||||||
|         return -4; |         return -4; | ||||||
|     } |     } | ||||||
|     cache = shm_rw; |     cache = shm_rw; | ||||||
| @@ -268,11 +269,11 @@ int cache_init() { | |||||||
| int cache_unload() { | int cache_unload() { | ||||||
|     int shm_id = shmget(CACHE_SHM_KEY, 0, 0); |     int shm_id = shmget(CACHE_SHM_KEY, 0, 0); | ||||||
|     if (shm_id < 0) { |     if (shm_id < 0) { | ||||||
|         fprintf(stderr, ERR_STR "Unable to get shared memory id: %s" CLR_STR "\n", strerror(errno)); |         fprintf(stderr, ERR_STR "Unable to get cache shared memory id: %s" CLR_STR "\n", strerror(errno)); | ||||||
|         shmdt(cache); |         shmdt(cache); | ||||||
|         return -1; |         return -1; | ||||||
|     } else if (shmctl(shm_id, IPC_RMID, NULL) < 0) { |     } else if (shmctl(shm_id, IPC_RMID, NULL) < 0) { | ||||||
|         fprintf(stderr, ERR_STR "Unable to configure shared memory: %s" CLR_STR "\n", strerror(errno)); |         fprintf(stderr, ERR_STR "Unable to configure cache shared memory: %s" CLR_STR "\n", strerror(errno)); | ||||||
|         shmdt(cache); |         shmdt(cache); | ||||||
|         return -1; |         return -1; | ||||||
|     } |     } | ||||||
| @@ -285,7 +286,7 @@ int cache_update_entry(int entry_num, const char *filename, const char *webroot) | |||||||
|     int shm_id = shmget(CACHE_SHM_KEY, 0, 0); |     int shm_id = shmget(CACHE_SHM_KEY, 0, 0); | ||||||
|     void *shm_rw = shmat(shm_id, NULL, 0); |     void *shm_rw = shmat(shm_id, NULL, 0); | ||||||
|     if (shm_rw == (void *) -1) { |     if (shm_rw == (void *) -1) { | ||||||
|         print(ERR_STR "Unable to attach shared memory (rw): %s" CLR_STR, strerror(errno)); |         print(ERR_STR "Unable to attach cache shared memory (rw): %s" CLR_STR, strerror(errno)); | ||||||
|         return -1; |         return -1; | ||||||
|     } |     } | ||||||
|     cache = shm_rw; |     cache = shm_rw; | ||||||
| @@ -328,7 +329,7 @@ int cache_filename_comp_invalid(const char *filename) { | |||||||
|     int shm_id = shmget(CACHE_SHM_KEY, 0, 0); |     int shm_id = shmget(CACHE_SHM_KEY, 0, 0); | ||||||
|     void *shm_rw = shmat(shm_id, NULL, 0); |     void *shm_rw = shmat(shm_id, NULL, 0); | ||||||
|     if (shm_rw == (void *) -1) { |     if (shm_rw == (void *) -1) { | ||||||
|         print(ERR_STR "Unable to attach shared memory (rw): %s" CLR_STR, strerror(errno)); |         print(ERR_STR "Unable to attach cache shared memory (rw): %s" CLR_STR, strerror(errno)); | ||||||
|         return -1; |         return -1; | ||||||
|     } |     } | ||||||
|     cache = shm_rw; |     cache = shm_rw; | ||||||
|   | |||||||
| @@ -6,6 +6,7 @@ | |||||||
|  */ |  */ | ||||||
|  |  | ||||||
| #include "compress.h" | #include "compress.h" | ||||||
|  |  | ||||||
| #include <malloc.h> | #include <malloc.h> | ||||||
| #include <errno.h> | #include <errno.h> | ||||||
|  |  | ||||||
|   | |||||||
| @@ -7,6 +7,7 @@ | |||||||
|  |  | ||||||
| #include "config.h" | #include "config.h" | ||||||
| #include "utils.h" | #include "utils.h" | ||||||
|  |  | ||||||
| #include <stdio.h> | #include <stdio.h> | ||||||
| #include <sys/ipc.h> | #include <sys/ipc.h> | ||||||
| #include <sys/shm.h> | #include <sys/shm.h> | ||||||
| @@ -20,20 +21,20 @@ char geoip_dir[256], dns_server[256]; | |||||||
| int config_init() { | int config_init() { | ||||||
|     int shm_id = shmget(CONFIG_SHM_KEY, sizeof(t_config), IPC_CREAT | IPC_EXCL | 0640); |     int shm_id = shmget(CONFIG_SHM_KEY, sizeof(t_config), IPC_CREAT | IPC_EXCL | 0640); | ||||||
|     if (shm_id < 0) { |     if (shm_id < 0) { | ||||||
|         fprintf(stderr, ERR_STR "Unable to create shared memory: %s" CLR_STR "\n", strerror(errno)); |         fprintf(stderr, ERR_STR "Unable to create config shared memory: %s" CLR_STR "\n", strerror(errno)); | ||||||
|         return -1; |         return -1; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     void *shm = shmat(shm_id, NULL, SHM_RDONLY); |     void *shm = shmat(shm_id, NULL, SHM_RDONLY); | ||||||
|     if (shm == (void *) -1) { |     if (shm == (void *) -1) { | ||||||
|         fprintf(stderr, ERR_STR "Unable to attach shared memory (ro): %s" CLR_STR "\n", strerror(errno)); |         fprintf(stderr, ERR_STR "Unable to attach config shared memory (ro): %s" CLR_STR "\n", strerror(errno)); | ||||||
|         return -2; |         return -2; | ||||||
|     } |     } | ||||||
|     config = shm; |     config = shm; | ||||||
|  |  | ||||||
|     void *shm_rw = shmat(shm_id, NULL, 0); |     void *shm_rw = shmat(shm_id, NULL, 0); | ||||||
|     if (shm_rw == (void *) -1) { |     if (shm_rw == (void *) -1) { | ||||||
|         fprintf(stderr, ERR_STR "Unable to attach shared memory (rw): %s" CLR_STR "\n", strerror(errno)); |         fprintf(stderr, ERR_STR "Unable to attach config shared memory (rw): %s" CLR_STR "\n", strerror(errno)); | ||||||
|         return -3; |         return -3; | ||||||
|     } |     } | ||||||
|     config = shm_rw; |     config = shm_rw; | ||||||
| @@ -46,11 +47,11 @@ int config_init() { | |||||||
| int config_unload() { | int config_unload() { | ||||||
|     int shm_id = shmget(CONFIG_SHM_KEY, 0, 0); |     int shm_id = shmget(CONFIG_SHM_KEY, 0, 0); | ||||||
|     if (shm_id < 0) { |     if (shm_id < 0) { | ||||||
|         fprintf(stderr, ERR_STR "Unable to get shared memory id: %s" CLR_STR "\n", strerror(errno)); |         fprintf(stderr, ERR_STR "Unable to get config shared memory id: %s" CLR_STR "\n", strerror(errno)); | ||||||
|         shmdt(config); |         shmdt(config); | ||||||
|         return -1; |         return -1; | ||||||
|     } else if (shmctl(shm_id, IPC_RMID, NULL) < 0) { |     } else if (shmctl(shm_id, IPC_RMID, NULL) < 0) { | ||||||
|         fprintf(stderr, ERR_STR "Unable to configure shared memory: %s" CLR_STR "\n", strerror(errno)); |         fprintf(stderr, ERR_STR "Unable to configure config shared memory: %s" CLR_STR "\n", strerror(errno)); | ||||||
|         shmdt(config); |         shmdt(config); | ||||||
|         return -1; |         return -1; | ||||||
|     } |     } | ||||||
| @@ -256,7 +257,7 @@ int config_load(const char *filename) { | |||||||
|  |  | ||||||
|     int shm_id = shmget(CONFIG_SHM_KEY, 0, 0); |     int shm_id = shmget(CONFIG_SHM_KEY, 0, 0); | ||||||
|     if (shm_id < 0) { |     if (shm_id < 0) { | ||||||
|         fprintf(stderr, ERR_STR "Unable to get shared memory id: %s" CLR_STR "\n", strerror(errno)); |         fprintf(stderr, ERR_STR "Unable to get config shared memory id: %s" CLR_STR "\n", strerror(errno)); | ||||||
|         shmdt(config); |         shmdt(config); | ||||||
|         return -3; |         return -3; | ||||||
|     } |     } | ||||||
| @@ -264,7 +265,7 @@ int config_load(const char *filename) { | |||||||
|     void *shm_rw = shmat(shm_id, NULL, 0); |     void *shm_rw = shmat(shm_id, NULL, 0); | ||||||
|     if (shm_rw == (void *) -1) { |     if (shm_rw == (void *) -1) { | ||||||
|         free(tmp_config); |         free(tmp_config); | ||||||
|         fprintf(stderr, ERR_STR "Unable to attach shared memory (rw): %s" CLR_STR "\n", strerror(errno)); |         fprintf(stderr, ERR_STR "Unable to attach config shared memory (rw): %s" CLR_STR "\n", strerror(errno)); | ||||||
|         return -4; |         return -4; | ||||||
|     } |     } | ||||||
|     memcpy(shm_rw, tmp_config, sizeof(t_config)); |     memcpy(shm_rw, tmp_config, sizeof(t_config)); | ||||||
|   | |||||||
| @@ -8,7 +8,8 @@ | |||||||
| #include "fastcgi.h" | #include "fastcgi.h" | ||||||
| #include "utils.h" | #include "utils.h" | ||||||
| #include "compress.h" | #include "compress.h" | ||||||
| #include "../necronda-server.h" | #include "../server.h" | ||||||
|  |  | ||||||
| #include <sys/un.h> | #include <sys/un.h> | ||||||
| #include <sys/socket.h> | #include <sys/socket.h> | ||||||
| #include <errno.h> | #include <errno.h> | ||||||
| @@ -41,9 +42,9 @@ char *fastcgi_add_param(char *buf, const char *key, const char *value) { | |||||||
|         ptr += 4; |         ptr += 4; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     memcpy(ptr, key, key_len); |     strcpy(ptr, key); | ||||||
|     ptr += key_len; |     ptr += key_len; | ||||||
|     memcpy(ptr, value, val_len); |     strcpy(ptr, value); | ||||||
|     ptr += val_len; |     ptr += val_len; | ||||||
|  |  | ||||||
|     return ptr; |     return ptr; | ||||||
| @@ -149,19 +150,21 @@ int fastcgi_init(fastcgi_conn *conn, int mode, unsigned int client_num, unsigned | |||||||
|     param_ptr = fastcgi_add_param(param_ptr, "PATH_INFO", buf0); |     param_ptr = fastcgi_add_param(param_ptr, "PATH_INFO", buf0); | ||||||
|  |  | ||||||
|     //param_ptr = fastcgi_add_param(param_ptr, "AUTH_TYPE", ""); |     //param_ptr = fastcgi_add_param(param_ptr, "AUTH_TYPE", ""); | ||||||
|     char *content_length = http_get_header_field(&req->hdr, "Content-Length"); |     const char *content_length = http_get_header_field(&req->hdr, "Content-Length"); | ||||||
|     param_ptr = fastcgi_add_param(param_ptr, "CONTENT_LENGTH", content_length != NULL ? content_length : ""); |     param_ptr = fastcgi_add_param(param_ptr, "CONTENT_LENGTH", content_length != NULL ? content_length : ""); | ||||||
|     char *content_type = http_get_header_field(&req->hdr, "Content-Type"); |     const char *content_type = http_get_header_field(&req->hdr, "Content-Type"); | ||||||
|     param_ptr = fastcgi_add_param(param_ptr, "CONTENT_TYPE", content_type != NULL ? content_type : ""); |     param_ptr = fastcgi_add_param(param_ptr, "CONTENT_TYPE", content_type != NULL ? content_type : ""); | ||||||
|     if (client_geoip != NULL) { |     if (client_geoip != NULL) { | ||||||
|         param_ptr = fastcgi_add_param(param_ptr, "REMOTE_INFO", client_geoip); |         param_ptr = fastcgi_add_param(param_ptr, "REMOTE_INFO", client_geoip); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     for (int i = 0; i < req->hdr.field_num; i++) { |     for (int i = 0; i < req->hdr.field_num; i++) { | ||||||
|  |         const http_field *f = &req->hdr.fields[i]; | ||||||
|  |         const char *name = http_field_get_name(f); | ||||||
|         char *ptr = buf0; |         char *ptr = buf0; | ||||||
|         ptr += sprintf(ptr, "HTTP_"); |         ptr += sprintf(ptr, "HTTP_"); | ||||||
|         for (int j = 0; j < strlen(req->hdr.fields[i][0]); j++, ptr++) { |         for (int j = 0; j < strlen(name); j++, ptr++) { | ||||||
|             char ch = req->hdr.fields[i][0][j]; |             char ch = name[j]; | ||||||
|             if ((ch >= 'A' && ch <= 'Z') || (ch >= '0' && ch <= '9')) { |             if ((ch >= 'A' && ch <= 'Z') || (ch >= '0' && ch <= '9')) { | ||||||
|                 ch = ch; |                 ch = ch; | ||||||
|             } else if (ch >= 'a' && ch <= 'z') { |             } else if (ch >= 'a' && ch <= 'z') { | ||||||
| @@ -172,7 +175,7 @@ int fastcgi_init(fastcgi_conn *conn, int mode, unsigned int client_num, unsigned | |||||||
|             ptr[0] = ch; |             ptr[0] = ch; | ||||||
|             ptr[1] = 0; |             ptr[1] = 0; | ||||||
|         } |         } | ||||||
|         param_ptr = fastcgi_add_param(param_ptr, buf0, req->hdr.fields[i][1]); |         param_ptr = fastcgi_add_param(param_ptr, buf0, http_field_get_value(f)); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     unsigned short param_len = param_ptr - param_buf - sizeof(header); |     unsigned short param_len = param_ptr - param_buf - sizeof(header); | ||||||
| @@ -219,7 +222,8 @@ int fastcgi_close_stdin(fastcgi_conn *conn) { | |||||||
| int fastcgi_php_error(const fastcgi_conn *conn, const char *msg, int msg_len, char *err_msg) { | int fastcgi_php_error(const fastcgi_conn *conn, const char *msg, int msg_len, char *err_msg) { | ||||||
|     char *msg_str = malloc(msg_len + 1); |     char *msg_str = malloc(msg_len + 1); | ||||||
|     char *ptr0 = msg_str; |     char *ptr0 = msg_str; | ||||||
|     strncpy(msg_str, msg, msg_len); |     memcpy(msg_str, msg, msg_len); | ||||||
|  |     msg_str[msg_len] = 0; | ||||||
|     char *ptr1 = NULL; |     char *ptr1 = NULL; | ||||||
|     int len; |     int len; | ||||||
|     int err = 0; |     int err = 0; | ||||||
| @@ -382,7 +386,7 @@ int fastcgi_header(fastcgi_conn *conn, http_res *res, char *err_msg) { | |||||||
|             return 1; |             return 1; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         ret = http_parse_header_field(&res->hdr, ptr, pos0); |         ret = http_parse_header_field(&res->hdr, ptr, pos0, 0); | ||||||
|         if (ret != 0) return (int) ret; |         if (ret != 0) return (int) ret; | ||||||
|         if (pos0[2] == '\r' && pos0[3] == '\n') { |         if (pos0[2] == '\r' && pos0[3] == '\n') { | ||||||
|             return 0; |             return 0; | ||||||
|   | |||||||
							
								
								
									
										237
									
								
								src/lib/http.c
									
									
									
									
									
								
							
							
						
						
									
										237
									
								
								src/lib/http.c
									
									
									
									
									
								
							| @@ -8,11 +8,14 @@ | |||||||
| #include "http.h" | #include "http.h" | ||||||
| #include "utils.h" | #include "utils.h" | ||||||
| #include "compress.h" | #include "compress.h" | ||||||
|  |  | ||||||
| #include <string.h> | #include <string.h> | ||||||
|  |  | ||||||
| void http_to_camel_case(char *str, int mode) { | void http_to_camel_case(char *str, int mode) { | ||||||
|     char last = '-'; |     if (mode == HTTP_PRESERVE) | ||||||
|     char ch; |         return; | ||||||
|  |  | ||||||
|  |     char ch, last = '-'; | ||||||
|     for (int i = 0; i < strlen(str); i++) { |     for (int i = 0; i < strlen(str); i++) { | ||||||
|         ch = str[i]; |         ch = str[i]; | ||||||
|         if (mode == HTTP_CAMEL && last == '-' && ch >= 'a' && ch <= 'z') { |         if (mode == HTTP_CAMEL && last == '-' && ch >= 'a' && ch <= 'z') { | ||||||
| @@ -24,12 +27,50 @@ void http_to_camel_case(char *str, int mode) { | |||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | const char *http_field_get_name(const http_field *field) { | ||||||
|  |     if (field->type == HTTP_FIELD_NORMAL) { | ||||||
|  |         return field->normal.name; | ||||||
|  |     } else if (field->type == HTTP_FIELD_EX_VALUE) { | ||||||
|  |         return field->ex_value.name; | ||||||
|  |     } else if (field->type == HTTP_FIELD_EX_NAME) { | ||||||
|  |         return field->ex_name.name; | ||||||
|  |     } | ||||||
|  |     return NULL; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | const char *http_field_get_value(const http_field *field) { | ||||||
|  |     if (field->type == HTTP_FIELD_NORMAL) { | ||||||
|  |         return field->normal.value; | ||||||
|  |     } else if (field->type == HTTP_FIELD_EX_VALUE) { | ||||||
|  |         return field->ex_value.value; | ||||||
|  |     } else if (field->type == HTTP_FIELD_EX_NAME) { | ||||||
|  |         return field->ex_name.value; | ||||||
|  |     } | ||||||
|  |     return NULL; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void http_free_field(http_field *f) { | ||||||
|  |     if (f->type == HTTP_FIELD_NORMAL) { | ||||||
|  |         f->normal.name[0] = 0; | ||||||
|  |         f->normal.value[0] = 0; | ||||||
|  |     } else if (f->type == HTTP_FIELD_EX_VALUE) { | ||||||
|  |         f->ex_value.name[0] = 0; | ||||||
|  |         free(f->ex_value.value); | ||||||
|  |         f->ex_value.value = NULL; | ||||||
|  |     } else if (f->type == HTTP_FIELD_EX_NAME) { | ||||||
|  |         free(f->ex_name.name); | ||||||
|  |         free(f->ex_name.value); | ||||||
|  |         f->ex_name.name = NULL; | ||||||
|  |         f->ex_name.value = NULL; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
| void http_free_hdr(http_hdr *hdr) { | void http_free_hdr(http_hdr *hdr) { | ||||||
|     for (int i = 0; i < hdr->field_num; i++) { |     for (int i = 0; i < hdr->field_num; i++) { | ||||||
|         free(hdr->fields[i][0]); |         http_free_field(&hdr->fields[i]); | ||||||
|         free(hdr->fields[i][1]); |  | ||||||
|     } |     } | ||||||
|     hdr->field_num = 0; |     hdr->field_num = 0; | ||||||
|  |     hdr->last_field_num = -1; | ||||||
| } | } | ||||||
|  |  | ||||||
| void http_free_req(http_req *req) { | void http_free_req(http_req *req) { | ||||||
| @@ -42,45 +83,64 @@ void http_free_res(http_res *res) { | |||||||
|     http_free_hdr(&res->hdr); |     http_free_hdr(&res->hdr); | ||||||
| } | } | ||||||
|  |  | ||||||
| int http_parse_header_field(http_hdr *hdr, const char *buf, const char *end_ptr) { | int http_parse_header_field(http_hdr *hdr, const char *buf, const char *end_ptr, int flags) { | ||||||
|     char *pos1 = memchr(buf, ':', end_ptr - buf); |     if (hdr->last_field_num > hdr->field_num) { | ||||||
|     char *pos2; |         print(ERR_STR "Unable to parse header: Invalid state" CLR_STR); | ||||||
|  |         return 3; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     char *pos1 = (char *) buf, *pos2 = (char *) end_ptr; | ||||||
|  |     if (buf[0] == ' ' || buf[0] == '\t') { | ||||||
|  |         if (hdr->last_field_num == -1) { | ||||||
|  |             print(ERR_STR "Unable to parse header" CLR_STR); | ||||||
|  |             return 3; | ||||||
|  |         } | ||||||
|  |         http_field *f = &hdr->fields[(int) hdr->last_field_num]; | ||||||
|  |  | ||||||
|  |         str_trim_lws(&pos1, &pos2); | ||||||
|  |         http_append_to_header_field(f, pos1, pos2 - pos1); | ||||||
|  |  | ||||||
|  |         return 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     pos1 = memchr(buf, ':', end_ptr - buf); | ||||||
|     if (pos1 == NULL) { |     if (pos1 == NULL) { | ||||||
|         print(ERR_STR "Unable to parse header" CLR_STR); |         print(ERR_STR "Unable to parse header" CLR_STR); | ||||||
|         return 3; |         return 3; | ||||||
|     } |     } | ||||||
|  |     long len1 = pos1 - buf; | ||||||
|     long len = pos1 - buf; |  | ||||||
|     hdr->fields[(int) hdr->field_num][0] = malloc(len + 1); |  | ||||||
|     sprintf(hdr->fields[(int) hdr->field_num][0], "%.*s", (int) len, buf); |  | ||||||
|     http_to_camel_case(hdr->fields[(int) hdr->field_num][0], HTTP_CAMEL); |  | ||||||
|  |  | ||||||
|     pos1++; |     pos1++; | ||||||
|     pos2 = (char *) end_ptr - 1; |     str_trim_lws(&pos1, &pos2); | ||||||
|     while (pos1[0] == ' ') pos1++; |     long len2 = pos2 - pos1; | ||||||
|     while (pos2[0] == ' ') pos2--; |  | ||||||
|     len = pos2 - pos1 + 1; |  | ||||||
|  |  | ||||||
|     if (len <= 0) { |     char field_num = hdr->field_num; | ||||||
|         hdr->fields[(int) hdr->field_num][1] = malloc(1); |     int found = http_get_header_field_num_len(hdr, buf, len1); | ||||||
|         hdr->fields[(int) hdr->field_num][1][0] = 0; |     if (!(flags & HTTP_MERGE_FIELDS) || found == -1) { | ||||||
|     } else { |         if (http_add_header_field_len(hdr, buf, len1, pos1, len2 < 0 ? 0 : len2) != 0) { | ||||||
|         hdr->fields[(int) hdr->field_num][1] = malloc(len + 1); |             print(ERR_STR "Unable to parse header: Too many header fields" CLR_STR); | ||||||
|         sprintf(hdr->fields[(int) hdr->field_num][1], "%.*s", (int) len, pos1); |             return 3; | ||||||
|         } |         } | ||||||
|     hdr->field_num++; |     } else { | ||||||
|  |         field_num = (char) found; | ||||||
|  |         http_append_to_header_field(&hdr->fields[found], ", ", 2); | ||||||
|  |         http_append_to_header_field(&hdr->fields[found], pos1, len2); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     hdr->last_field_num = (char) field_num; | ||||||
|     return 0; |     return 0; | ||||||
| } | } | ||||||
|  |  | ||||||
| int http_receive_request(sock *client, http_req *req) { | int http_receive_request(sock *client, http_req *req) { | ||||||
|     long rcv_len, len; |     long rcv_len, len; | ||||||
|     char *ptr, *pos0, *pos1, *pos2; |  | ||||||
|     char buf[CLIENT_MAX_HEADER_SIZE]; |     char buf[CLIENT_MAX_HEADER_SIZE]; | ||||||
|  |     char *ptr, *pos0 = buf, *pos1, *pos2; | ||||||
|     memset(buf, 0, sizeof(buf)); |     memset(buf, 0, sizeof(buf)); | ||||||
|     memset(req->method, 0, sizeof(req->method)); |     memset(req->method, 0, sizeof(req->method)); | ||||||
|     memset(req->version, 0, sizeof(req->version)); |     memset(req->version, 0, sizeof(req->version)); | ||||||
|     req->uri = NULL; |     req->uri = NULL; | ||||||
|     req->hdr.field_num = 0; |     req->hdr.field_num = 0; | ||||||
|  |     req->hdr.last_field_num = -1; | ||||||
|  |  | ||||||
|     while (1) { |     while (1) { | ||||||
|         rcv_len  = sock_recv(client, buf, CLIENT_MAX_HEADER_SIZE, 0); |         rcv_len  = sock_recv(client, buf, CLIENT_MAX_HEADER_SIZE, 0); | ||||||
| @@ -144,7 +204,7 @@ int http_receive_request(sock *client, http_req *req) { | |||||||
|                 sprintf(req->uri, "%.*s", (int) len, pos1); |                 sprintf(req->uri, "%.*s", (int) len, pos1); | ||||||
|                 sprintf(req->version, "%.3s", pos2 + 5); |                 sprintf(req->version, "%.3s", pos2 + 5); | ||||||
|             } else { |             } else { | ||||||
|                 int ret = http_parse_header_field(&req->hdr, ptr, pos0); |                 int ret = http_parse_header_field(&req->hdr, ptr, pos0, HTTP_MERGE_FIELDS); | ||||||
|                 if (ret != 0) return ret; |                 if (ret != 0) return ret; | ||||||
|             } |             } | ||||||
|             ptr = pos0 + 2; |             ptr = pos0 + 2; | ||||||
| @@ -164,31 +224,95 @@ int http_receive_request(sock *client, http_req *req) { | |||||||
|     return 0; |     return 0; | ||||||
| } | } | ||||||
|  |  | ||||||
| char *http_get_header_field(const http_hdr *hdr, const char *field_name) { | const char *http_get_header_field(const http_hdr *hdr, const char *field_name) { | ||||||
|     char field_name_1[256], field_name_2[256]; |     return http_get_header_field_len(hdr, field_name, strlen(field_name)); | ||||||
|     strcpy(field_name_1, field_name); | } | ||||||
|     http_to_camel_case(field_name_1, HTTP_LOWER); |  | ||||||
|     for (int i = 0; i < hdr->field_num; i++) { | const char *http_get_header_field_len(const http_hdr *hdr, const char *field_name, unsigned long len) { | ||||||
|         strcpy(field_name_2, hdr->fields[i][0]); |     int num = http_get_header_field_num_len(hdr, field_name, len); | ||||||
|         http_to_camel_case(field_name_2, HTTP_LOWER); |     return (num >= 0 && num < HTTP_MAX_HEADER_FIELD_NUM) ? http_field_get_value(&hdr->fields[num]) : NULL; | ||||||
|         if (strcmp(field_name_1, field_name_2) == 0) { | } | ||||||
|             return hdr->fields[i][1]; |  | ||||||
|         } | int http_get_header_field_num(const http_hdr *hdr, const char *field_name) { | ||||||
|     } |     return http_get_header_field_num_len(hdr, field_name, strlen(field_name)); | ||||||
|     return NULL; | } | ||||||
|  |  | ||||||
|  | int http_get_header_field_num_len(const http_hdr *hdr, const char *field_name, unsigned long len) { | ||||||
|  |     char field_name_1[256], field_name_2[256]; | ||||||
|  |     memcpy(field_name_1, field_name, len); | ||||||
|  |     field_name_1[len] = 0; | ||||||
|  |     http_to_camel_case(field_name_1, HTTP_LOWER); | ||||||
|  |  | ||||||
|  |     for (int i = 0; i < hdr->field_num; i++) { | ||||||
|  |         strcpy(field_name_2, http_field_get_name(&hdr->fields[i])); | ||||||
|  |         http_to_camel_case(field_name_2, HTTP_LOWER); | ||||||
|  |  | ||||||
|  |         if (strcmp(field_name_1, field_name_2) == 0) | ||||||
|  |             return i; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return -1; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | int http_add_header_field(http_hdr *hdr, const char *field_name, const char *field_value) { | ||||||
|  |     return http_add_header_field_len(hdr, field_name, strlen(field_name), field_value, strlen(field_value)); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | int http_add_header_field_len(http_hdr *hdr, const char *name, unsigned long name_len, const char *value, unsigned long value_len) { | ||||||
|  |     if (hdr->field_num >= HTTP_MAX_HEADER_FIELD_NUM) | ||||||
|  |         return -1; | ||||||
|  |  | ||||||
|  |     http_field *f = &hdr->fields[(int) hdr->field_num]; | ||||||
|  |  | ||||||
|  |     if (name_len < sizeof(f->normal.name) && value_len < sizeof(f->normal.value)) { | ||||||
|  |         f->type = HTTP_FIELD_NORMAL; | ||||||
|  |         memcpy(f->normal.name, name, name_len); | ||||||
|  |         memcpy(f->normal.value, value, value_len); | ||||||
|  |         f->normal.name[name_len] = 0; | ||||||
|  |         f->normal.value[value_len] = 0; | ||||||
|  |         http_to_camel_case(f->normal.name, HTTP_PRESERVE); | ||||||
|  |     } else if (name_len < sizeof(f->ex_value.name)) { | ||||||
|  |         f->type = HTTP_FIELD_EX_VALUE; | ||||||
|  |         f->ex_value.value = malloc(value_len + 1); | ||||||
|  |         memcpy(f->ex_value.name, name, name_len); | ||||||
|  |         memcpy(f->ex_value.value, value, value_len); | ||||||
|  |         f->ex_value.name[name_len] = 0; | ||||||
|  |         f->ex_value.value[value_len] = 0; | ||||||
|  |         http_to_camel_case(f->ex_value.name, HTTP_PRESERVE); | ||||||
|  |     } else { | ||||||
|  |         f->type = HTTP_FIELD_EX_NAME; | ||||||
|  |         f->ex_name.name = malloc(name_len + 1); | ||||||
|  |         f->ex_name.value = malloc(value_len + 1); | ||||||
|  |         memcpy(f->ex_name.name, name, name_len); | ||||||
|  |         memcpy(f->ex_name.value, value, value_len); | ||||||
|  |         f->ex_name.name[name_len] = 0; | ||||||
|  |         f->ex_name.value[value_len] = 0; | ||||||
|  |         http_to_camel_case(f->ex_name.name, HTTP_PRESERVE); | ||||||
|     } |     } | ||||||
|  |  | ||||||
| void http_add_header_field(http_hdr *hdr, const char *field_name, const char *field_value) { |  | ||||||
|     size_t len_name = strlen(field_name); |  | ||||||
|     size_t len_value = strlen(field_value); |  | ||||||
|     char *_field_name = malloc(len_name + 1); |  | ||||||
|     char *_field_value = malloc(len_value + 1); |  | ||||||
|     strcpy(_field_name, field_name); |  | ||||||
|     strcpy(_field_value, field_value); |  | ||||||
|     http_to_camel_case(_field_name, HTTP_PRESERVE); |  | ||||||
|     hdr->fields[(int) hdr->field_num][0] = _field_name; |  | ||||||
|     hdr->fields[(int) hdr->field_num][1] = _field_value; |  | ||||||
|     hdr->field_num++; |     hdr->field_num++; | ||||||
|  |     return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void http_append_to_header_field(http_field *field, const char *value, unsigned long len) { | ||||||
|  |     if (field->type == HTTP_FIELD_NORMAL) { | ||||||
|  |         unsigned long total_len = strlen(field->normal.value) + len + 1; | ||||||
|  |         if (total_len < sizeof(field->normal.value)) { | ||||||
|  |             strncat(field->normal.value, value, len); | ||||||
|  |         } else { | ||||||
|  |             field->type = HTTP_FIELD_EX_VALUE; | ||||||
|  |             char *new = malloc(total_len); | ||||||
|  |             strcpy(new, field->normal.value); | ||||||
|  |             strncat(new, value, len); | ||||||
|  |             field->ex_value.value = new; | ||||||
|  |         } | ||||||
|  |     } else if (field->type == HTTP_FIELD_EX_VALUE) { | ||||||
|  |         field->ex_value.value = realloc(field->ex_value.value, strlen(field->ex_value.value) + len + 1); | ||||||
|  |         strncat(field->ex_value.value, value, len); | ||||||
|  |     } else if (field->type == HTTP_FIELD_EX_NAME) { | ||||||
|  |         field->ex_name.value = realloc(field->ex_name.value, strlen(field->ex_name.value) + len + 1); | ||||||
|  |         strncat(field->ex_name.value, value, len); | ||||||
|  |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| void http_remove_header_field(http_hdr *hdr, const char *field_name, int mode) { | void http_remove_header_field(http_hdr *hdr, const char *field_name, int mode) { | ||||||
| @@ -203,12 +327,11 @@ void http_remove_header_field(http_hdr *hdr, const char *field_name, int mode) { | |||||||
|         diff = -1; |         diff = -1; | ||||||
|     } |     } | ||||||
|     for (; i < hdr->field_num && i >= 0; i += diff) { |     for (; i < hdr->field_num && i >= 0; i += diff) { | ||||||
|         strcpy(field_name_2, hdr->fields[i][0]); |         strcpy(field_name_2, http_field_get_name(&hdr->fields[i])); | ||||||
|         http_to_camel_case(field_name_2, HTTP_LOWER); |         http_to_camel_case(field_name_2, HTTP_LOWER); | ||||||
|         if (strcmp(field_name_1, field_name_2) == 0) { |         if (strcmp(field_name_1, field_name_2) == 0) { | ||||||
|             for (int j = i; j < hdr->field_num - 1; j++) { |             http_free_field(&hdr->fields[i]); | ||||||
|                 memcpy(hdr->fields[j], hdr->fields[j + 1], sizeof(hdr->fields[0])); |             memmove(&hdr->fields[i], &hdr->fields[i + 1], sizeof(hdr->fields[0]) * (hdr->field_num - i)); | ||||||
|             } |  | ||||||
|             hdr->field_num--; |             hdr->field_num--; | ||||||
|             if (mode == HTTP_REMOVE_ALL) { |             if (mode == HTTP_REMOVE_ALL) { | ||||||
|                 i -= diff; |                 i -= diff; | ||||||
| @@ -223,7 +346,8 @@ int http_send_response(sock *client, http_res *res) { | |||||||
|     char buf[CLIENT_MAX_HEADER_SIZE]; |     char buf[CLIENT_MAX_HEADER_SIZE]; | ||||||
|     long off = sprintf(buf, "HTTP/%s %03i %s\r\n", res->version, res->status->code, res->status->msg); |     long off = sprintf(buf, "HTTP/%s %03i %s\r\n", res->version, res->status->code, res->status->msg); | ||||||
|     for (int i = 0; i < res->hdr.field_num; i++) { |     for (int i = 0; i < res->hdr.field_num; i++) { | ||||||
|         off += sprintf(buf + off, "%s: %s\r\n", res->hdr.fields[i][0], res->hdr.fields[i][1]); |         const http_field *f = &res->hdr.fields[i]; | ||||||
|  |         off += sprintf(buf + off, "%s: %s\r\n", http_field_get_name(f), http_field_get_value(f)); | ||||||
|     } |     } | ||||||
|     off += sprintf(buf + off, "\r\n"); |     off += sprintf(buf + off, "\r\n"); | ||||||
|     if (sock_send(client, buf, off, 0) < 0) { |     if (sock_send(client, buf, off, 0) < 0) { | ||||||
| @@ -236,7 +360,8 @@ int http_send_request(sock *server, http_req *req) { | |||||||
|     char buf[CLIENT_MAX_HEADER_SIZE]; |     char buf[CLIENT_MAX_HEADER_SIZE]; | ||||||
|     long off = sprintf(buf, "%s %s HTTP/%s\r\n", req->method, req->uri, req->version); |     long off = sprintf(buf, "%s %s HTTP/%s\r\n", req->method, req->uri, req->version); | ||||||
|     for (int i = 0; i < req->hdr.field_num; i++) { |     for (int i = 0; i < req->hdr.field_num; i++) { | ||||||
|         off += sprintf(buf + off, "%s: %s\r\n", req->hdr.fields[i][0], req->hdr.fields[i][1]); |         const http_field *f = &req->hdr.fields[i]; | ||||||
|  |         off += sprintf(buf + off, "%s: %s\r\n", http_field_get_name(f), http_field_get_value(f)); | ||||||
|     } |     } | ||||||
|     off += sprintf(buf + off, "\r\n"); |     off += sprintf(buf + off, "\r\n"); | ||||||
|     long ret = sock_send(server, buf, off, 0); |     long ret = sock_send(server, buf, off, 0); | ||||||
| @@ -314,9 +439,9 @@ const http_doc_info *http_get_status_info(const http_status *status) { | |||||||
| } | } | ||||||
|  |  | ||||||
| int http_get_compression(const http_req *req, const http_res *res) { | int http_get_compression(const http_req *req, const http_res *res) { | ||||||
|     char *accept_encoding = http_get_header_field(&req->hdr, "Accept-Encoding"); |     const char *accept_encoding = http_get_header_field(&req->hdr, "Accept-Encoding"); | ||||||
|     char *content_type = http_get_header_field(&res->hdr, "Content-Type"); |     const char *content_type = http_get_header_field(&res->hdr, "Content-Type"); | ||||||
|     char *content_encoding = http_get_header_field(&res->hdr, "Content-Encoding"); |     const char *content_encoding = http_get_header_field(&res->hdr, "Content-Encoding"); | ||||||
|     if (mime_is_compressible(content_type) && content_encoding == NULL && accept_encoding != NULL) { |     if (mime_is_compressible(content_type) && content_encoding == NULL && accept_encoding != NULL) { | ||||||
|         if (strstr(accept_encoding, "br") != NULL) { |         if (strstr(accept_encoding, "br") != NULL) { | ||||||
|             return COMPRESS_BR; |             return COMPRESS_BR; | ||||||
|   | |||||||
| @@ -18,6 +18,12 @@ | |||||||
| #define HTTP_REMOVE_ALL 1 | #define HTTP_REMOVE_ALL 1 | ||||||
| #define HTTP_REMOVE_LAST 2 | #define HTTP_REMOVE_LAST 2 | ||||||
|  |  | ||||||
|  | #define HTTP_FIELD_NORMAL 0 | ||||||
|  | #define HTTP_FIELD_EX_VALUE 1 | ||||||
|  | #define HTTP_FIELD_EX_NAME 2 | ||||||
|  |  | ||||||
|  | #define HTTP_MERGE_FIELDS 1 | ||||||
|  |  | ||||||
| #define HTTP_1XX_STR "\x1B[1;32m" | #define HTTP_1XX_STR "\x1B[1;32m" | ||||||
| #define HTTP_2XX_STR "\x1B[1;32m" | #define HTTP_2XX_STR "\x1B[1;32m" | ||||||
| #define HTTP_3XX_STR "\x1B[1;33m" | #define HTTP_3XX_STR "\x1B[1;33m" | ||||||
| @@ -30,6 +36,7 @@ | |||||||
| #define HTTP_COLOR_ERROR "#C00000" | #define HTTP_COLOR_ERROR "#C00000" | ||||||
|  |  | ||||||
| #define CLIENT_MAX_HEADER_SIZE 8192 | #define CLIENT_MAX_HEADER_SIZE 8192 | ||||||
|  | #define HTTP_MAX_HEADER_FIELD_NUM 64 | ||||||
|  |  | ||||||
| #ifndef SERVER_STR | #ifndef SERVER_STR | ||||||
| #   define SERVER_STR "Necronda" | #   define SERVER_STR "Necronda" | ||||||
| @@ -57,9 +64,28 @@ typedef struct { | |||||||
|     const char *doc; |     const char *doc; | ||||||
| } http_doc_info; | } http_doc_info; | ||||||
|  |  | ||||||
|  | typedef struct { | ||||||
|  |     char type; | ||||||
|  |     union { | ||||||
|  |         struct { | ||||||
|  |             char name[64]; | ||||||
|  |             char value[192]; | ||||||
|  |         } normal; | ||||||
|  |         struct { | ||||||
|  |             char name[192]; | ||||||
|  |             char *value; | ||||||
|  |         } ex_value; | ||||||
|  |         struct { | ||||||
|  |             char *name; | ||||||
|  |             char *value; | ||||||
|  |         } ex_name; | ||||||
|  |     }; | ||||||
|  | } http_field; | ||||||
|  |  | ||||||
| typedef struct { | typedef struct { | ||||||
|     char field_num; |     char field_num; | ||||||
|     char *fields[64][2]; |     char last_field_num; | ||||||
|  |     http_field fields[HTTP_MAX_HEADER_FIELD_NUM]; | ||||||
| } http_hdr; | } http_hdr; | ||||||
|  |  | ||||||
| typedef struct { | typedef struct { | ||||||
| @@ -102,6 +128,12 @@ extern const char http_info_icon[]; | |||||||
|  |  | ||||||
| void http_to_camel_case(char *str, int mode); | void http_to_camel_case(char *str, int mode); | ||||||
|  |  | ||||||
|  | const char *http_field_get_name(const http_field *field); | ||||||
|  |  | ||||||
|  | const char *http_field_get_value(const http_field *field); | ||||||
|  |  | ||||||
|  | void http_free_field(http_field *f); | ||||||
|  |  | ||||||
| void http_free_hdr(http_hdr *hdr); | void http_free_hdr(http_hdr *hdr); | ||||||
|  |  | ||||||
| void http_free_req(http_req *req); | void http_free_req(http_req *req); | ||||||
| @@ -110,11 +142,21 @@ void http_free_res(http_res *res); | |||||||
|  |  | ||||||
| int http_receive_request(sock *client, http_req *req); | int http_receive_request(sock *client, http_req *req); | ||||||
|  |  | ||||||
| int http_parse_header_field(http_hdr *hdr, const char *buf, const char *end_ptr) ; | int http_parse_header_field(http_hdr *hdr, const char *buf, const char *end_ptr, int flags); | ||||||
|  |  | ||||||
| char *http_get_header_field(const http_hdr *hdr, const char *field_name); | const char *http_get_header_field(const http_hdr *hdr, const char *field_name); | ||||||
|  |  | ||||||
| void http_add_header_field(http_hdr *hdr, const char *field_name, const char *field_value); | const char *http_get_header_field_len(const http_hdr *hdr, const char *field_name, unsigned long len); | ||||||
|  |  | ||||||
|  | int http_get_header_field_num(const http_hdr *hdr, const char *field_name); | ||||||
|  |  | ||||||
|  | int http_get_header_field_num_len(const http_hdr *hdr, const char *field_name, unsigned long len); | ||||||
|  |  | ||||||
|  | int http_add_header_field(http_hdr *hdr, const char *field_name, const char *field_value); | ||||||
|  |  | ||||||
|  | int http_add_header_field_len(http_hdr *hdr, const char *name, unsigned long name_len, const char *value, unsigned long value_len); | ||||||
|  |  | ||||||
|  | void http_append_to_header_field(http_field *field, const char *value, unsigned long len); | ||||||
|  |  | ||||||
| void http_remove_header_field(http_hdr *hdr, const char *field_name, int mode); | void http_remove_header_field(http_hdr *hdr, const char *field_name, int mode); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -8,7 +8,8 @@ | |||||||
| #include "rev_proxy.h" | #include "rev_proxy.h" | ||||||
| #include "utils.h" | #include "utils.h" | ||||||
| #include "compress.h" | #include "compress.h" | ||||||
| #include "../necronda-server.h" | #include "../server.h" | ||||||
|  |  | ||||||
| #include <openssl/ssl.h> | #include <openssl/ssl.h> | ||||||
| #include <string.h> | #include <string.h> | ||||||
| #include <errno.h> | #include <errno.h> | ||||||
| @@ -29,13 +30,12 @@ int rev_proxy_preload() { | |||||||
| } | } | ||||||
|  |  | ||||||
| int rev_proxy_request_header(http_req *req, int enc) { | int rev_proxy_request_header(http_req *req, int enc) { | ||||||
|     char buf1[256]; |     char buf1[256], buf2[256]; | ||||||
|     char buf2[256]; |  | ||||||
|     int p_len; |     int p_len; | ||||||
|     http_remove_header_field(&req->hdr, "Connection", HTTP_REMOVE_ALL); |     http_remove_header_field(&req->hdr, "Connection", HTTP_REMOVE_ALL); | ||||||
|     http_add_header_field(&req->hdr, "Connection", "keep-alive"); |     http_add_header_field(&req->hdr, "Connection", "keep-alive"); | ||||||
|  |  | ||||||
|     char *via = http_get_header_field(&req->hdr, "Via"); |     const char *via = http_get_header_field(&req->hdr, "Via"); | ||||||
|     sprintf(buf1, "HTTP/%s %s", req->version, SERVER_NAME); |     sprintf(buf1, "HTTP/%s %s", req->version, SERVER_NAME); | ||||||
|     if (via == NULL) { |     if (via == NULL) { | ||||||
|         http_add_header_field(&req->hdr, "Via", buf1); |         http_add_header_field(&req->hdr, "Via", buf1); | ||||||
| @@ -49,8 +49,8 @@ int rev_proxy_request_header(http_req *req, int enc) { | |||||||
|         http_add_header_field(&req->hdr, "Via", buf2); |         http_add_header_field(&req->hdr, "Via", buf2); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     char *host = http_get_header_field(&req->hdr, "Host"); |     const char *host = http_get_header_field(&req->hdr, "Host"); | ||||||
|     char *forwarded = http_get_header_field(&req->hdr, "Forwarded"); |     const char *forwarded = http_get_header_field(&req->hdr, "Forwarded"); | ||||||
|     int client_ipv6 = strchr(client_addr_str, ':') != NULL; |     int client_ipv6 = strchr(client_addr_str, ':') != NULL; | ||||||
|     int server_ipv6 = strchr(server_addr_str, ':') != NULL; |     int server_ipv6 = strchr(server_addr_str, ':') != NULL; | ||||||
|  |  | ||||||
| @@ -62,6 +62,7 @@ int rev_proxy_request_header(http_req *req, int enc) { | |||||||
|         print(ERR_STR "Appended part of header field 'Forwarded' too long" CLR_STR); |         print(ERR_STR "Appended part of header field 'Forwarded' too long" CLR_STR); | ||||||
|         return -1; |         return -1; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     if (forwarded == NULL) { |     if (forwarded == NULL) { | ||||||
|         http_add_header_field(&req->hdr, "Forwarded", buf1); |         http_add_header_field(&req->hdr, "Forwarded", buf1); | ||||||
|     } else { |     } else { | ||||||
| @@ -74,7 +75,7 @@ int rev_proxy_request_header(http_req *req, int enc) { | |||||||
|         http_add_header_field(&req->hdr, "Forwarded", buf2); |         http_add_header_field(&req->hdr, "Forwarded", buf2); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     char *xff = http_get_header_field(&req->hdr, "X-Forwarded-For"); |     const char *xff = http_get_header_field(&req->hdr, "X-Forwarded-For"); | ||||||
|     if (xff == NULL) { |     if (xff == NULL) { | ||||||
|         http_add_header_field(&req->hdr, "X-Forwarded-For", client_addr_str); |         http_add_header_field(&req->hdr, "X-Forwarded-For", client_addr_str); | ||||||
|     } else { |     } else { | ||||||
| @@ -83,7 +84,7 @@ int rev_proxy_request_header(http_req *req, int enc) { | |||||||
|         http_add_header_field(&req->hdr, "X-Forwarded-For", buf1); |         http_add_header_field(&req->hdr, "X-Forwarded-For", buf1); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     char *xfh = http_get_header_field(&req->hdr, "X-Forwarded-Host"); |     const char *xfh = http_get_header_field(&req->hdr, "X-Forwarded-Host"); | ||||||
|     if (xfh == NULL) { |     if (xfh == NULL) { | ||||||
|         if (forwarded == NULL) { |         if (forwarded == NULL) { | ||||||
|             http_add_header_field(&req->hdr, "X-Forwarded-Host", host); |             http_add_header_field(&req->hdr, "X-Forwarded-Host", host); | ||||||
| @@ -104,7 +105,7 @@ int rev_proxy_request_header(http_req *req, int enc) { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     char *xfp = http_get_header_field(&req->hdr, "X-Forwarded-Proto"); |     const char *xfp = http_get_header_field(&req->hdr, "X-Forwarded-Proto"); | ||||||
|     if (xfp == NULL) { |     if (xfp == NULL) { | ||||||
|         if (forwarded == NULL) { |         if (forwarded == NULL) { | ||||||
|             http_add_header_field(&req->hdr, "X-Forwarded-Proto", enc ? "https" : "http"); |             http_add_header_field(&req->hdr, "X-Forwarded-Proto", enc ? "https" : "http"); | ||||||
| @@ -128,12 +129,11 @@ int rev_proxy_request_header(http_req *req, int enc) { | |||||||
|     return 0; |     return 0; | ||||||
| } | } | ||||||
|  |  | ||||||
| int rev_proxy_response_header(http_req *req, http_res *res) { | int rev_proxy_response_header(http_req *req, http_res *res, host_config *conf) { | ||||||
|     char buf1[256]; |     char buf1[256], buf2[256]; | ||||||
|     char buf2[256]; |  | ||||||
|     int p_len; |     int p_len; | ||||||
|  |  | ||||||
|     char *via = http_get_header_field(&res->hdr, "Via"); |     const char *via = http_get_header_field(&res->hdr, "Via"); | ||||||
|     p_len = snprintf(buf1, sizeof(buf1), "HTTP/%s %s", req->version, SERVER_NAME); |     p_len = snprintf(buf1, sizeof(buf1), "HTTP/%s %s", req->version, SERVER_NAME); | ||||||
|     if (p_len < 0 || p_len >= sizeof(buf1)) { |     if (p_len < 0 || p_len >= sizeof(buf1)) { | ||||||
|         print(ERR_STR "Appended part of header field 'Via' too long" CLR_STR); |         print(ERR_STR "Appended part of header field 'Via' too long" CLR_STR); | ||||||
| @@ -151,15 +151,40 @@ int rev_proxy_response_header(http_req *req, http_res *res) { | |||||||
|         http_add_header_field(&res->hdr, "Via", buf2); |         http_add_header_field(&res->hdr, "Via", buf2); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     const char *location = http_get_header_field(&res->hdr, "Location"); | ||||||
|  |     if (location != NULL) { | ||||||
|  |         char *hostnames[] = {conf->name, conf->rev_proxy.hostname}; | ||||||
|  |         for (int i = 0; i < sizeof(hostnames) / sizeof(hostnames[0]); i++) { | ||||||
|  |             char *hostname = hostnames[i]; | ||||||
|  |  | ||||||
|  |             p_len = snprintf(buf1, sizeof(buf1), "http://%s/", hostname); | ||||||
|  |             if (strncmp(location, buf1, p_len) == 0) goto match; | ||||||
|  |  | ||||||
|  |             p_len = snprintf(buf1, sizeof(buf1), "https://%s/", hostname); | ||||||
|  |             if (strncmp(location, buf1, p_len) == 0) goto match; | ||||||
|  |  | ||||||
|  |             p_len = snprintf(buf1, sizeof(buf1), "http://%s:%i/", hostname, conf->rev_proxy.port); | ||||||
|  |             if (strncmp(location, buf1, p_len) == 0) goto match; | ||||||
|  |  | ||||||
|  |             p_len = snprintf(buf1, sizeof(buf1), "https://%s:%i/", hostname, conf->rev_proxy.port); | ||||||
|  |             if (strncmp(location, buf1, p_len) == 0) goto match; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (0) { | ||||||
|  |             match: | ||||||
|  |             strcpy(buf1, location + p_len - 1); | ||||||
|  |             http_remove_header_field(&res->hdr, "Location", HTTP_REMOVE_ALL); | ||||||
|  |             http_add_header_field(&res->hdr, "Location", buf1); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|     return 0; |     return 0; | ||||||
| } | } | ||||||
|  |  | ||||||
| int rev_proxy_init(http_req *req, http_res *res, http_status_ctx *ctx, host_config *conf, sock *client, | int rev_proxy_init(http_req *req, http_res *res, http_status_ctx *ctx, host_config *conf, sock *client, http_status *custom_status, char *err_msg) { | ||||||
|                    http_status *custom_status, char *err_msg) { |  | ||||||
|     char buffer[CHUNK_SIZE]; |     char buffer[CHUNK_SIZE]; | ||||||
|     long ret; |     long ret; | ||||||
|     int tries = 0; |     int tries = 0, retry = 0; | ||||||
|     int retry = 0; |  | ||||||
|  |  | ||||||
|     if (rev_proxy.socket != 0 && strcmp(rev_proxy_host, conf->name) == 0 && sock_check(&rev_proxy) == 0) { |     if (rev_proxy.socket != 0 && strcmp(rev_proxy_host, conf->name) == 0 && sock_check(&rev_proxy) == 0) { | ||||||
|         goto rev_proxy; |         goto rev_proxy; | ||||||
| @@ -281,7 +306,7 @@ int rev_proxy_init(http_req *req, http_res *res, http_status_ctx *ctx, host_conf | |||||||
|         goto proxy_err; |         goto proxy_err; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     char *content_length = http_get_header_field(&req->hdr, "Content-Length"); |     const char *content_length = http_get_header_field(&req->hdr, "Content-Length"); | ||||||
|     if (content_length != NULL) { |     if (content_length != NULL) { | ||||||
|         unsigned long content_len = strtoul(content_length, NULL, 10); |         unsigned long content_len = strtoul(content_length, NULL, 10); | ||||||
|         if (client->buf_len - client->buf_off > 0) { |         if (client->buf_len - client->buf_off > 0) { | ||||||
| @@ -338,6 +363,7 @@ int rev_proxy_init(http_req *req, http_res *res, http_status_ctx *ctx, host_conf | |||||||
|         } |         } | ||||||
|         print(ERR_STR "Unable to receive response from server: %s" CLR_STR, sock_strerror(&rev_proxy)); |         print(ERR_STR "Unable to receive response from server: %s" CLR_STR, sock_strerror(&rev_proxy)); | ||||||
|         sprintf(err_msg, "Unable to receive response from server: %s.", sock_strerror(&rev_proxy)); |         sprintf(err_msg, "Unable to receive response from server: %s.", sock_strerror(&rev_proxy)); | ||||||
|  |         retry = tries < 4; | ||||||
|         goto proxy_err; |         goto proxy_err; | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -396,7 +422,7 @@ int rev_proxy_init(http_req *req, http_res *res, http_status_ctx *ctx, host_conf | |||||||
|                 goto proxy_err; |                 goto proxy_err; | ||||||
|             } |             } | ||||||
|         } else { |         } else { | ||||||
|             ret = http_parse_header_field(&res->hdr, ptr, pos0); |             ret = http_parse_header_field(&res->hdr, ptr, pos0, 0); | ||||||
|             if (ret != 0) { |             if (ret != 0) { | ||||||
|                 res->status = http_get_status(502); |                 res->status = http_get_status(502); | ||||||
|                 ctx->origin = SERVER_RES; |                 ctx->origin = SERVER_RES; | ||||||
| @@ -412,7 +438,7 @@ int rev_proxy_init(http_req *req, http_res *res, http_status_ctx *ctx, host_conf | |||||||
|     } |     } | ||||||
|     sock_recv(&rev_proxy, buffer, header_len, 0); |     sock_recv(&rev_proxy, buffer, header_len, 0); | ||||||
|  |  | ||||||
|     ret = rev_proxy_response_header(req, res); |     ret = rev_proxy_response_header(req, res, conf); | ||||||
|     if (ret != 0) { |     if (ret != 0) { | ||||||
|         res->status = http_get_status(500); |         res->status = http_get_status(500); | ||||||
|         ctx->origin = INTERNAL; |         ctx->origin = INTERNAL; | ||||||
| @@ -428,13 +454,9 @@ int rev_proxy_init(http_req *req, http_res *res, http_status_ctx *ctx, host_conf | |||||||
|  |  | ||||||
| int rev_proxy_send(sock *client, unsigned long len_to_send, int flags) { | int rev_proxy_send(sock *client, unsigned long len_to_send, int flags) { | ||||||
|     // TODO handle websockets |     // TODO handle websockets | ||||||
|     long ret; |     char buffer[CHUNK_SIZE], comp_out[CHUNK_SIZE], buf[256], *ptr; | ||||||
|     char buffer[CHUNK_SIZE]; |     long ret = 0, len, snd_len; | ||||||
|     char comp_out[CHUNK_SIZE]; |  | ||||||
|     char buf[256]; |  | ||||||
|     long len, snd_len; |  | ||||||
|     int finish_comp = 0; |     int finish_comp = 0; | ||||||
|     char *ptr; |  | ||||||
|  |  | ||||||
|     compress_ctx comp_ctx; |     compress_ctx comp_ctx; | ||||||
|     if (flags & REV_PROXY_COMPRESS_BR) { |     if (flags & REV_PROXY_COMPRESS_BR) { | ||||||
| @@ -471,6 +493,7 @@ int rev_proxy_send(sock *client, unsigned long len_to_send, int flags) { | |||||||
|             if (len_to_send == 0 && (flags & REV_PROXY_COMPRESS)) { |             if (len_to_send == 0 && (flags & REV_PROXY_COMPRESS)) { | ||||||
|                 finish_comp = 1; |                 finish_comp = 1; | ||||||
|                 len = 0; |                 len = 0; | ||||||
|  |                 ptr = NULL; | ||||||
|                 goto out; |                 goto out; | ||||||
|                 finish: |                 finish: | ||||||
|                 compress_free(&comp_ctx); |                 compress_free(&comp_ctx); | ||||||
| @@ -492,8 +515,7 @@ int rev_proxy_send(sock *client, unsigned long len_to_send, int flags) { | |||||||
|                 long buf_len = len; |                 long buf_len = len; | ||||||
|                 if (flags & REV_PROXY_COMPRESS) { |                 if (flags & REV_PROXY_COMPRESS) { | ||||||
|                     avail_out = sizeof(comp_out); |                     avail_out = sizeof(comp_out); | ||||||
|                     compress_compress(&comp_ctx, next_in + len - avail_in, &avail_in, comp_out, &avail_out, |                     compress_compress(&comp_ctx, next_in + len - avail_in, &avail_in, comp_out, &avail_out, finish_comp); | ||||||
|                                       finish_comp); |  | ||||||
|                     ptr = comp_out; |                     ptr = comp_out; | ||||||
|                     buf_len = (int) (sizeof(comp_out) - avail_out); |                     buf_len = (int) (sizeof(comp_out) - avail_out); | ||||||
|                     snd_len += (long) (len - avail_in); |                     snd_len += (long) (len - avail_in); | ||||||
|   | |||||||
| @@ -26,7 +26,7 @@ int rev_proxy_preload(); | |||||||
|  |  | ||||||
| int rev_proxy_request_header(http_req *req, int enc); | int rev_proxy_request_header(http_req *req, int enc); | ||||||
|  |  | ||||||
| int rev_proxy_response_header(http_req *req, http_res *res); | int rev_proxy_response_header(http_req *req, http_res *res, host_config *conf); | ||||||
|  |  | ||||||
| int rev_proxy_init(http_req *req, http_res *res, http_status_ctx *ctx, host_config *conf, sock *client, | int rev_proxy_init(http_req *req, http_res *res, http_status_ctx *ctx, host_config *conf, sock *client, | ||||||
|                    http_status *custom_status, char *err_msg); |                    http_status *custom_status, char *err_msg); | ||||||
|   | |||||||
| @@ -6,6 +6,7 @@ | |||||||
|  */ |  */ | ||||||
|  |  | ||||||
| #include "sock.h" | #include "sock.h" | ||||||
|  |  | ||||||
| #include <openssl/err.h> | #include <openssl/err.h> | ||||||
| #include <openssl/ssl.h> | #include <openssl/ssl.h> | ||||||
| #include <string.h> | #include <string.h> | ||||||
|   | |||||||
| @@ -7,6 +7,7 @@ | |||||||
|  |  | ||||||
| #include "uri.h" | #include "uri.h" | ||||||
| #include "utils.h" | #include "utils.h" | ||||||
|  |  | ||||||
| #include <stdlib.h> | #include <stdlib.h> | ||||||
| #include <string.h> | #include <string.h> | ||||||
|  |  | ||||||
|   | |||||||
| @@ -6,6 +6,7 @@ | |||||||
|  */ |  */ | ||||||
|  |  | ||||||
| #include "utils.h" | #include "utils.h" | ||||||
|  |  | ||||||
| #include <stdio.h> | #include <stdio.h> | ||||||
| #include <string.h> | #include <string.h> | ||||||
| #include <stdlib.h> | #include <stdlib.h> | ||||||
| @@ -124,7 +125,6 @@ int mime_is_compressible(const char *type) { | |||||||
|         strcmp(type_parsed, "application/vnd.ms-fontobject") == 0 || |         strcmp(type_parsed, "application/vnd.ms-fontobject") == 0 || | ||||||
|         strcmp(type_parsed, "application/x-font-ttf") == 0 || |         strcmp(type_parsed, "application/x-font-ttf") == 0 || | ||||||
|         strcmp(type_parsed, "application/x-javascript") == 0 || |         strcmp(type_parsed, "application/x-javascript") == 0 || | ||||||
|         strcmp(type_parsed, "application/x-web-app-manifest+json") == 0 || |  | ||||||
|         strcmp(type_parsed, "font/eot") == 0 || |         strcmp(type_parsed, "font/eot") == 0 || | ||||||
|         strcmp(type_parsed, "font/opentype") == 0 || |         strcmp(type_parsed, "font/opentype") == 0 || | ||||||
|         strcmp(type_parsed, "image/bmp") == 0 || |         strcmp(type_parsed, "image/bmp") == 0 || | ||||||
| @@ -135,11 +135,62 @@ int mime_is_compressible(const char *type) { | |||||||
| } | } | ||||||
|  |  | ||||||
| int strcpy_rem_webroot(char *dst, const char *src, long len, const char *webroot) { | int strcpy_rem_webroot(char *dst, const char *src, long len, const char *webroot) { | ||||||
|     strncpy(dst, src, len); |     memcpy(dst, src, len); | ||||||
|     if (webroot == NULL) return 0; |     dst[len] = 0; | ||||||
|  |     if (webroot == NULL) | ||||||
|  |         return 0; | ||||||
|  |  | ||||||
|     char *pos; |     char *pos; | ||||||
|  |     const unsigned long webroot_len = strlen(webroot); | ||||||
|  |     if (webroot_len == 0) | ||||||
|  |         return 0; | ||||||
|  |  | ||||||
|     while ((pos = strstr(dst, webroot)) != NULL) { |     while ((pos = strstr(dst, webroot)) != NULL) { | ||||||
|         strcpy(pos, pos + strlen(webroot)); |         strcpy(pos, pos + webroot_len); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | int str_trim(char **start, char **end) { | ||||||
|  |     if (start == NULL || end == NULL || *start == NULL || *end == NULL) | ||||||
|  |         return -1; | ||||||
|  |  | ||||||
|  |     (*end)--; | ||||||
|  |     while (*start[0] == ' ' || *start[0] == '\t' || *start[0] == '\r' || *start[0] == '\n') (*start)++; | ||||||
|  |     while (*end[0] == ' ' || *end[0] == '\t' || *end[0] == '\r' || *end[0] == '\n') (*end)--; | ||||||
|  |     (*end)++; | ||||||
|  |     return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | int str_trim_lws(char **start, char **end) { | ||||||
|  |     if (start == NULL || end == NULL || *start == NULL || *end == NULL) | ||||||
|  |         return -1; | ||||||
|  |  | ||||||
|  |     (*end)--; | ||||||
|  |     while (*start[0] == ' ' || *start[0] == '\t') (*start)++; | ||||||
|  |     while (*end[0] == ' ' || *end[0] == '\t') (*end)--; | ||||||
|  |     (*end)++; | ||||||
|  |     return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | int base64_encode(void *data, unsigned long data_len, char *output, unsigned long *output_len) { | ||||||
|  |     unsigned long out_len = 4 * ((data_len + 2) / 3); | ||||||
|  |     if (output_len != NULL) *output_len = out_len; | ||||||
|  |  | ||||||
|  |     for (int i = 0, j = 0; i < data_len;) { | ||||||
|  |         unsigned int octet_a = (i < data_len) ? ((unsigned char *) data)[i++] : 0; | ||||||
|  |         unsigned int octet_b = (i < data_len) ? ((unsigned char *) data)[i++] : 0; | ||||||
|  |         unsigned int octet_c = (i < data_len) ? ((unsigned char *) data)[i++] : 0; | ||||||
|  |         unsigned int triple = (octet_a << 0x10) + (octet_b << 0x08) + octet_c; | ||||||
|  |         output[j++] = base64_encode_table[(triple >> 3 * 6) & 0x3F]; | ||||||
|  |         output[j++] = base64_encode_table[(triple >> 2 * 6) & 0x3F]; | ||||||
|  |         output[j++] = base64_encode_table[(triple >> 1 * 6) & 0x3F]; | ||||||
|  |         output[j++] = base64_encode_table[(triple >> 0 * 6) & 0x3F]; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     for (int i = 0; i < base64_mod_table[data_len % 3]; i++) | ||||||
|  |         output[out_len - 1 - i] = '='; | ||||||
|  |  | ||||||
|     return 0; |     return 0; | ||||||
| } | } | ||||||
|   | |||||||
| @@ -20,6 +20,10 @@ | |||||||
|  |  | ||||||
| extern char *log_prefix; | extern char *log_prefix; | ||||||
|  |  | ||||||
|  | static const char base64_encode_table[64] = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/"; | ||||||
|  | static const int base64_mod_table[3] = {0, 2, 1}; | ||||||
|  |  | ||||||
|  |  | ||||||
| #define out_1(fmt) fprintf(stdout, "%s" fmt "\n", log_prefix) | #define out_1(fmt) fprintf(stdout, "%s" fmt "\n", log_prefix) | ||||||
| #define out_2(fmt, args...) fprintf(stdout, "%s" fmt "\n", log_prefix, args) | #define out_2(fmt, args...) fprintf(stdout, "%s" fmt "\n", log_prefix, args) | ||||||
|  |  | ||||||
| @@ -42,4 +46,10 @@ int mime_is_compressible(const char *type); | |||||||
|  |  | ||||||
| int strcpy_rem_webroot(char *dst, const char *str, long len, const char *webroot); | int strcpy_rem_webroot(char *dst, const char *str, long len, const char *webroot); | ||||||
|  |  | ||||||
|  | int str_trim(char **start, char **end); | ||||||
|  |  | ||||||
|  | int str_trim_lws(char **start, char **end); | ||||||
|  |  | ||||||
|  | int base64_encode(void *data, unsigned long data_len, char *output, unsigned long *output_len); | ||||||
|  |  | ||||||
| #endif //NECRONDA_SERVER_UTILS_H | #endif //NECRONDA_SERVER_UTILS_H | ||||||
|   | |||||||
| @@ -8,14 +8,15 @@ | |||||||
| #define _POSIX_C_SOURCE 199309L | #define _POSIX_C_SOURCE 199309L | ||||||
| 
 | 
 | ||||||
| #include "necronda.h" | #include "necronda.h" | ||||||
| #include "necronda-server.h" | #include "server.h" | ||||||
| #include "client.c" | #include "client.h" | ||||||
| 
 | 
 | ||||||
| #include "lib/cache.h" | #include "lib/cache.h" | ||||||
| #include "lib/config.h" | #include "lib/config.h" | ||||||
| #include "lib/sock.h" | #include "lib/sock.h" | ||||||
| #include "lib/rev_proxy.h" | #include "lib/rev_proxy.h" | ||||||
| #include "lib/geoip.h" | #include "lib/geoip.h" | ||||||
|  | #include "lib/utils.h" | ||||||
| 
 | 
 | ||||||
| #include <stdio.h> | #include <stdio.h> | ||||||
| #include <sys/socket.h> | #include <sys/socket.h> | ||||||
| @@ -65,13 +66,11 @@ void destroy() { | |||||||
|         if (children[i] != 0) { |         if (children[i] != 0) { | ||||||
|             ret = waitpid(children[i], &status, WNOHANG); |             ret = waitpid(children[i], &status, WNOHANG); | ||||||
|             if (ret < 0) { |             if (ret < 0) { | ||||||
|                 fprintf(stderr, ERR_STR "Unable to wait for child process (PID %i): %s" CLR_STR "\n", |                 fprintf(stderr, ERR_STR "Unable to wait for child process (PID %i): %s" CLR_STR "\n", children[i], strerror(errno)); | ||||||
|                         children[i], strerror(errno)); |  | ||||||
|             } else if (ret == children[i]) { |             } else if (ret == children[i]) { | ||||||
|                 children[i] = 0; |                 children[i] = 0; | ||||||
|                 if (status != 0) { |                 if (status != 0) { | ||||||
|                     fprintf(stderr, ERR_STR "Child process with PID %i terminated with exit code %i" CLR_STR "\n", |                     fprintf(stderr, ERR_STR "Child process with PID %i terminated with exit code %i" CLR_STR "\n", ret, status); | ||||||
|                             ret, status); |  | ||||||
|                 } |                 } | ||||||
|             } else { |             } else { | ||||||
|                 kill(children[i], SIGKILL); |                 kill(children[i], SIGKILL); | ||||||
| @@ -106,13 +105,11 @@ void terminate() { | |||||||
|         if (children[i] != 0) { |         if (children[i] != 0) { | ||||||
|             ret = waitpid(children[i], &status, WNOHANG); |             ret = waitpid(children[i], &status, WNOHANG); | ||||||
|             if (ret < 0) { |             if (ret < 0) { | ||||||
|                 fprintf(stderr, ERR_STR "Unable to wait for child process (PID %i): %s" CLR_STR "\n", |                 fprintf(stderr, ERR_STR "Unable to wait for child process (PID %i): %s" CLR_STR "\n", children[i], strerror(errno)); | ||||||
|                         children[i], strerror(errno)); |  | ||||||
|             } else if (ret == children[i]) { |             } else if (ret == children[i]) { | ||||||
|                 children[i] = 0; |                 children[i] = 0; | ||||||
|                 if (status != 0) { |                 if (status != 0) { | ||||||
|                     fprintf(stderr, ERR_STR "Child process with PID %i terminated with exit code %i" CLR_STR "\n", |                     fprintf(stderr, ERR_STR "Child process with PID %i terminated with exit code %i" CLR_STR "\n", ret, status); | ||||||
|                             ret, status); |  | ||||||
|                 } |                 } | ||||||
|             } else { |             } else { | ||||||
|                 kill(children[i], SIGTERM); |                 kill(children[i], SIGTERM); | ||||||
| @@ -129,13 +126,11 @@ void terminate() { | |||||||
|         if (children[i] != 0) { |         if (children[i] != 0) { | ||||||
|             ret = waitpid(children[i], &status, 0); |             ret = waitpid(children[i], &status, 0); | ||||||
|             if (ret < 0) { |             if (ret < 0) { | ||||||
|                 fprintf(stderr, ERR_STR "Unable to wait for child process (PID %i): %s" CLR_STR "\n", |                 fprintf(stderr, ERR_STR "Unable to wait for child process (PID %i): %s" CLR_STR "\n", children[i], strerror(errno)); | ||||||
|                         children[i], strerror(errno)); |  | ||||||
|             } else if (ret == children[i]) { |             } else if (ret == children[i]) { | ||||||
|                 children[i] = 0; |                 children[i] = 0; | ||||||
|                 if (status != 0) { |                 if (status != 0) { | ||||||
|                     fprintf(stderr, ERR_STR "Child process with PID %i terminated with exit code %i" CLR_STR "\n", |                     fprintf(stderr, ERR_STR "Child process with PID %i terminated with exit code %i" CLR_STR "\n", ret, status); | ||||||
|                             ret, status); |  | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
| @@ -316,15 +311,15 @@ int main(int argc, const char *argv[]) { | |||||||
|         SSL_CTX_set_tlsext_servername_callback(ctx, ssl_servername_cb); |         SSL_CTX_set_tlsext_servername_callback(ctx, ssl_servername_cb); | ||||||
| 
 | 
 | ||||||
|         if (SSL_CTX_use_certificate_chain_file(ctx, conf->full_chain) != 1) { |         if (SSL_CTX_use_certificate_chain_file(ctx, conf->full_chain) != 1) { | ||||||
|             fprintf(stderr, ERR_STR "Unable to load certificate chain file: %s: %s" CLR_STR "\n", |             fprintf(stderr, ERR_STR "Unable to load certificate chain file: %s: %s" CLR_STR "\n", ERR_reason_error_string(ERR_get_error()), conf->full_chain); | ||||||
|                     ERR_reason_error_string(ERR_get_error()), conf->full_chain); |  | ||||||
|             config_unload(); |             config_unload(); | ||||||
|  |             cache_unload(); | ||||||
|             return 1; |             return 1; | ||||||
|         } |         } | ||||||
|         if (SSL_CTX_use_PrivateKey_file(ctx, conf->priv_key, SSL_FILETYPE_PEM) != 1) { |         if (SSL_CTX_use_PrivateKey_file(ctx, conf->priv_key, SSL_FILETYPE_PEM) != 1) { | ||||||
|             fprintf(stderr, ERR_STR "Unable to load private key file: %s: %s" CLR_STR "\n", |             fprintf(stderr, ERR_STR "Unable to load private key file: %s: %s" CLR_STR "\n", ERR_reason_error_string(ERR_get_error()), conf->priv_key); | ||||||
|                     ERR_reason_error_string(ERR_get_error()), conf->priv_key); |  | ||||||
|             config_unload(); |             config_unload(); | ||||||
|  |             cache_unload(); | ||||||
|             return 1; |             return 1; | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| @@ -338,6 +333,7 @@ int main(int argc, const char *argv[]) { | |||||||
|         if (listen(sockets[i], LISTEN_BACKLOG) < 0) { |         if (listen(sockets[i], LISTEN_BACKLOG) < 0) { | ||||||
|             fprintf(stderr, ERR_STR "Unable to listen on socket %i: %s" CLR_STR "\n", i, strerror(errno)); |             fprintf(stderr, ERR_STR "Unable to listen on socket %i: %s" CLR_STR "\n", i, strerror(errno)); | ||||||
|             config_unload(); |             config_unload(); | ||||||
|  |             cache_unload(); | ||||||
|             return 1; |             return 1; | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| @@ -378,7 +374,7 @@ int main(int argc, const char *argv[]) { | |||||||
|                     signal(SIGTERM, SIG_IGN); |                     signal(SIGTERM, SIG_IGN); | ||||||
| 
 | 
 | ||||||
|                     client.socket = client_fd; |                     client.socket = client_fd; | ||||||
|                     client.enc = i == 1; |                     client.enc = (i == 1); | ||||||
|                     return client_handler(&client, client_num, &client_addr); |                     return client_handler(&client, client_num, &client_addr); | ||||||
|                 } else if (pid > 0) { |                 } else if (pid > 0) { | ||||||
|                     // parent
 |                     // parent
 | ||||||
| @@ -402,18 +398,18 @@ int main(int argc, const char *argv[]) { | |||||||
|             if (children[i] != 0) { |             if (children[i] != 0) { | ||||||
|                 ret = waitpid(children[i], &status, WNOHANG); |                 ret = waitpid(children[i], &status, WNOHANG); | ||||||
|                 if (ret < 0) { |                 if (ret < 0) { | ||||||
|                     fprintf(stderr, ERR_STR "Unable to wait for child process (PID %i): %s" CLR_STR "\n", |                     fprintf(stderr, ERR_STR "Unable to wait for child process (PID %i): %s" CLR_STR "\n", children[i], strerror(errno)); | ||||||
|                             children[i], strerror(errno)); |  | ||||||
|                 } else if (ret == children[i]) { |                 } else if (ret == children[i]) { | ||||||
|                     children[i] = 0; |                     children[i] = 0; | ||||||
|                     if (status != 0) { |                     if (status != 0) { | ||||||
|                         fprintf(stderr, ERR_STR "Child process with PID %i terminated with exit code %i" CLR_STR "\n", |                         fprintf(stderr, ERR_STR "Child process with PID %i terminated with exit code %i" CLR_STR "\n", ret, status); | ||||||
|                                 ret, status); |  | ||||||
|                     } |                     } | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     config_unload(); | ||||||
|  |     cache_unload(); | ||||||
|     return 0; |     return 0; | ||||||
| } | } | ||||||
| @@ -5,8 +5,8 @@ | |||||||
|  * Lorenz Stechauner, 2020-12-03 |  * Lorenz Stechauner, 2020-12-03 | ||||||
|  */ |  */ | ||||||
| 
 | 
 | ||||||
| #ifndef NECRONDA_SERVER_NECRONDA_SERVER_H | #ifndef NECRONDA_SERVER_SERVER_H | ||||||
| #define NECRONDA_SERVER_NECRONDA_SERVER_H | #define NECRONDA_SERVER_SERVER_H | ||||||
| 
 | 
 | ||||||
| #include <sys/time.h> | #include <sys/time.h> | ||||||
| #include <maxminddb.h> | #include <maxminddb.h> | ||||||
| @@ -31,4 +31,4 @@ extern char *log_client_prefix, *log_conn_prefix, *log_req_prefix, *client_geoip | |||||||
| extern char *client_addr_str, *client_addr_str_ptr, *server_addr_str, *server_addr_str_ptr, *client_host_str; | extern char *client_addr_str, *client_addr_str_ptr, *server_addr_str, *server_addr_str_ptr, *client_host_str; | ||||||
| extern struct timeval client_timeout; | extern struct timeval client_timeout; | ||||||
| 
 | 
 | ||||||
| #endif //NECRONDA_SERVER_NECRONDA_SERVER_H
 | #endif //NECRONDA_SERVER_SERVER_H
 | ||||||
		Reference in New Issue
	
	Block a user