Compare commits
	
		
			37 Commits
		
	
	
		
			v4.2
			...
			10d405e745
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 10d405e745 | |||
| 28f163f97a | |||
| a8914aa981 | |||
| 2ada22481d | |||
| 17b25a3596 | |||
| d130474989 | |||
| 8dea0cd3fc | |||
| 0406cad0d8 | |||
| 21b7ab585a | |||
| 7587e15749 | |||
| 1b44752f91 | |||
| 5eeb9ef3c1 | |||
| dd15b9d906 | |||
| 12922a0661 | |||
| c0799101b1 | |||
| c1d076db04 | |||
| 10464f3f30 | |||
| 63781472fa | |||
| 81931d287d | |||
| 531ddb4880 | |||
| e0d8ab31d5 | |||
| 3a36d54e9d | |||
| 33d9aa3a5d | |||
| f4d30206b0 | |||
| 5b094ba98d | |||
| f60cdc8228 | |||
| ab1c4d6fd4 | |||
| 26d54e9968 | |||
| b6c7d8f58e | |||
| 53fcceeafb | |||
| 96567909db | |||
| 4b3c067a75 | |||
| 77b80ca67b | |||
| 70e76d8783 | |||
| 6b1bc54cf3 | |||
| e1edb48a3c | |||
| dc5d1bebcc | 
							
								
								
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -1,7 +1,8 @@ | ||||
| * | ||||
| !src | ||||
| !src/** | ||||
| !run.sh | ||||
| !docs | ||||
| !docs/** | ||||
| !Makefile | ||||
| !.gitignore | ||||
| !README.md | ||||
|   | ||||
							
								
								
									
										27
									
								
								Makefile
									
									
									
									
									
								
							
							
						
						
									
										27
									
								
								Makefile
									
									
									
									
									
								
							| @@ -1,19 +1,28 @@ | ||||
| .DEFAULT_GOAL := install | ||||
|  | ||||
| CFLAGS=-std=c11 -Wall | ||||
| INCLUDE=-lssl -lcrypto -lmagic -lz -lmaxminddb | ||||
| LIBS=src/lib/*.c | ||||
|  | ||||
| DEBIAN_OPTS=-D MAGIC_FILE="\"/usr/share/file/magic.mgc\"" -D PHP_FPM_SOCKET="\"/var/run/php/php7.3-fpm.sock\"" | ||||
|  | ||||
| packages: | ||||
| 	@echo "Installing packages..." | ||||
| 	sudo apt-get install gcc libmagic-dev libssl-dev php-fpm | ||||
| 	sudo apt install gcc php-fpm libmagic-dev libssl-dev libmaxminddb-dev | ||||
| 	@echo "Finished downloading!" | ||||
|  | ||||
| permit: | ||||
| 	sudo setcap 'cap_net_bind_service=+ep' "$(shell pwd)/bin/necronda-server" | ||||
|  | ||||
| compile: | ||||
| 	@mkdir -p bin | ||||
| 	gcc src/necronda-server.c -o bin/necronda-server -std=c11 -lssl -lcrypto -lmagic -lz -lmaxminddb | ||||
| 	gcc $(LIBS) -o bin/libnecronda-server.so --shared -fPIC $(CFLAGS) $(INCLUDE) | ||||
| 	gcc src/necronda-server.c -o bin/necronda-server $(CFLAGS) $(INCLUDE) \ | ||||
| 		-Lbin -lnecronda-server -Wl,-rpath=$(shell pwd)/bin | ||||
|  | ||||
| compile-debian: | ||||
| 	@mkdir -p bin | ||||
| 	gcc src/necronda-server.c -o bin/necronda-server -std=c11 -lssl -lcrypto -lmagic -lz -lmaxminddb \ | ||||
| 		-D MAGIC_FILE="\"/usr/share/file/magic.mgc\"" \ | ||||
| 		-D PHP_FPM_SOCKET="\"/var/run/php/php7.3-fpm.sock\"" | ||||
|  | ||||
| install: | packages compile | ||||
| 	@echo "Finished!" | ||||
| 	gcc $(LIBS) -o bin/libnecronda-server.so --shared -fPIC $(CFLAGS) $(INCLUDE) \ | ||||
| 		$(DEBIAN_OPTS) | ||||
| 	gcc src/necronda-server.c -o bin/necronda-server $(CFLAGS) $(INCLUDE) \ | ||||
| 		-Lbin -lnecronda-server -Wl,-rpath=$(shell pwd)/bin \ | ||||
| 		$(DEBIAN_OPTS) | ||||
|   | ||||
							
								
								
									
										44
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										44
									
								
								README.md
									
									
									
									
									
								
							| @@ -1,2 +1,44 @@ | ||||
|  | ||||
| # Necronda web server | ||||
| Necronda web server | ||||
| =================== | ||||
|  | ||||
| ## Features | ||||
|  | ||||
| * Full IPv4 and IPv6 support | ||||
| * Serving local files via HTTP and HTTPS | ||||
|   * File compression and disk cache for compressed files | ||||
| * Reverse proxy for other HTTP and HTTPS servers | ||||
| * FastCGI support (e.g. [PHP-FPM](https://php-fpm.org/)) | ||||
| * Support for [MaxMind's GeoIP Database](https://www.maxmind.com/en/geoip2-services-and-databases) | ||||
| * DNS reverse lookup for connecting hosts | ||||
| * Automatic URL rewrite (e.g. `/index.html` -> `/`, `/test.php` -> `/test`) | ||||
| * Modern looking and responsive error documents | ||||
|  | ||||
|  | ||||
| ## Configuration | ||||
|  | ||||
| See [docs/example.conf](docs/example.conf) for more details. | ||||
|  | ||||
|  | ||||
| ### Global directives | ||||
|  | ||||
| * `certificate` - path to SSL certificate (or certificate chain) | ||||
| * `private_key` - path to SSL private key | ||||
| * `geoip_dir` (optional) - path to a directory containing GeoIP databases | ||||
| * `dns_server` (optional) - address of a DNS server | ||||
|  | ||||
|  | ||||
| ### Virtual host configuration | ||||
|  | ||||
| * `[<host>]` - begins section for the virtual host `<host>` | ||||
| * Local | ||||
|     * `webroot` - path to the root directory | ||||
|     * `dir_mode` - specify the behaviour for directories without an `index.html` or `index.php` | ||||
|         * `forbidden` - the server will respond with `403 Forbidden` | ||||
|         * `info` - try passing *path info* to an upper `.php` file. | ||||
|         * `list` - list contents of directory (**not implemented yet**) | ||||
| * Reverse proxy | ||||
|     * `hostname` - hostname of server to be reverse proxy of | ||||
|     * `port` - port to be used | ||||
|     * `http` - use HTTP to communicate with server | ||||
|     * `https` - use HTTPS to communicate with server | ||||
|   | ||||
							
								
								
									
										19
									
								
								docs/example.conf
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								docs/example.conf
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,19 @@ | ||||
|  | ||||
| certificate /var/cert/cert.pem | ||||
| private_key /var/cert/cert.key | ||||
| #geoip_dir  /var/dir | ||||
| #dns_server 192.168.0.1 | ||||
|  | ||||
| [localhost] | ||||
| webroot     /var/www/localhost | ||||
| dir_mode    forbidden | ||||
|  | ||||
| [me.local] | ||||
| hostname    www.example.com | ||||
| port        80 | ||||
| http | ||||
|  | ||||
| [secure.local] | ||||
| hostname    www.example.com | ||||
| port        443 | ||||
| https | ||||
							
								
								
									
										6
									
								
								run.sh
									
									
									
									
									
								
							
							
						
						
									
										6
									
								
								run.sh
									
									
									
									
									
								
							| @@ -1,6 +0,0 @@ | ||||
| #!/bin/bash | ||||
| echo "-- Building and starting Necronda Server..." | ||||
| make compile && \ | ||||
|  echo "-- Successfully finished compiling!" && \ | ||||
|  echo "-- Starting Server..." && \ | ||||
|  ./bin/necronda-server $@ | ||||
							
								
								
									
										161
									
								
								src/client.c
									
									
									
									
									
								
							
							
						
						
									
										161
									
								
								src/client.c
									
									
									
									
									
								
							| @@ -5,18 +5,32 @@ | ||||
|  * Lorenz Stechauner, 2020-12-03 | ||||
|  */ | ||||
|  | ||||
| #include "necronda-server.h" | ||||
| #include "utils.h" | ||||
| #include "uri.h" | ||||
| #include "http.h" | ||||
| #include "fastcgi.h" | ||||
| #include "client.h" | ||||
| #include "lib/utils.h" | ||||
| #include "lib/config.h" | ||||
| #include "lib/sock.h" | ||||
| #include "lib/http.h" | ||||
| #include "lib/rev_proxy.h" | ||||
| #include "lib/fastcgi.h" | ||||
| #include "lib/cache.h" | ||||
|  | ||||
| #include <string.h> | ||||
| #include <sys/select.h> | ||||
| #include <errno.h> | ||||
| #include <unistd.h> | ||||
| #include <openssl/ssl.h> | ||||
| #include <openssl/err.h> | ||||
| #include <signal.h> | ||||
| #include <arpa/inet.h> | ||||
|  | ||||
| int server_keep_alive = 1; | ||||
| char *log_client_prefix, *log_conn_prefix, *log_req_prefix, *client_geoip; | ||||
|  | ||||
| struct timeval client_timeout = {.tv_sec = CLIENT_TIMEOUT, .tv_usec = 0}; | ||||
|  | ||||
| int server_keep_alive; | ||||
| char *log_client_prefix, *log_conn_prefix, *log_req_prefix, *client_geoip; | ||||
| char *client_addr_str, *client_addr_str_ptr, *server_addr_str, *server_addr_str_ptr, *client_host_str; | ||||
| struct timeval client_timeout; | ||||
|  | ||||
| host_config *get_host_config(const char *host) { | ||||
|     for (int i = 0; i < MAX_HOST_CONFIG; i++) { | ||||
|         host_config *hc = &config[i]; | ||||
| @@ -51,6 +65,7 @@ int client_request_handler(sock *client, unsigned long client_num, unsigned int | ||||
|     int accept_if_modified_since = 0; | ||||
|     int use_fastcgi = 0; | ||||
|     int use_rev_proxy = 0; | ||||
|     int p_len; | ||||
|     fastcgi_conn php_fpm = {.socket = 0, .req_id = 0}; | ||||
|     http_status custom_status; | ||||
|  | ||||
| @@ -58,8 +73,6 @@ int client_request_handler(sock *client, unsigned long client_num, unsigned int | ||||
|     sprintf(res.version, "1.1"); | ||||
|     res.status = http_get_status(501); | ||||
|     res.hdr.field_num = 0; | ||||
|     http_add_header_field(&res.hdr, "Date", http_get_date(buf0, sizeof(buf0))); | ||||
|     http_add_header_field(&res.hdr, "Server", SERVER_STR); | ||||
|  | ||||
|     clock_gettime(CLOCK_MONOTONIC, &begin); | ||||
|  | ||||
| @@ -69,6 +82,8 @@ int client_request_handler(sock *client, unsigned long client_num, unsigned int | ||||
|     client_timeout.tv_sec = CLIENT_TIMEOUT; | ||||
|     client_timeout.tv_usec = 0; | ||||
|     ret = select(client->socket + 1, &socket_fds, NULL, NULL, &client_timeout); | ||||
|     http_add_header_field(&res.hdr, "Date", http_get_date(buf0, sizeof(buf0))); | ||||
|     http_add_header_field(&res.hdr, "Server", SERVER_STR); | ||||
|     if (ret <= 0) { | ||||
|         if (errno != 0) { | ||||
|             return 1; | ||||
| @@ -102,7 +117,7 @@ int client_request_handler(sock *client, unsigned long client_num, unsigned int | ||||
|  | ||||
|     hdr_connection = http_get_header_field(&req.hdr, "Connection"); | ||||
|     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"); | ||||
|     if (host_ptr != NULL && strlen(host_ptr) > 255) { | ||||
|         host[0] = 0; | ||||
| @@ -141,10 +156,16 @@ int client_request_handler(sock *client, unsigned long client_num, unsigned int | ||||
|     if (ret != 0) { | ||||
|         if (ret == 1) { | ||||
|             sprintf(err_msg, "Invalid URI: has to start with slash."); | ||||
|             res.status = http_get_status(400); | ||||
|         } else if (ret == 2) { | ||||
|             sprintf(err_msg, "Invalid URI: contains relative path change (/../)."); | ||||
|             res.status = http_get_status(400); | ||||
|         } else if (ret == 3) { | ||||
|             sprintf(err_msg, "The specified webroot directory does not exist."); | ||||
|             res.status = http_get_status(404); | ||||
|         } else { | ||||
|             res.status = http_get_status(500); | ||||
|         } | ||||
|         res.status = http_get_status(400); | ||||
|         goto respond; | ||||
|     } | ||||
|  | ||||
| @@ -155,19 +176,53 @@ int client_request_handler(sock *client, unsigned long client_num, unsigned int | ||||
|         if (strcmp(uri.uri, buf0) != 0 || change_proto) { | ||||
|             res.status = http_get_status(308); | ||||
|             size = sizeof(buf0); | ||||
|             encode_url(uri.uri, buf0, &size); | ||||
|             url_encode(uri.uri, buf0, &size); | ||||
|             if (change_proto) { | ||||
|                 sprintf(buf1, "https://%s%s", host, buf0); | ||||
|                 p_len = snprintf(buf1, sizeof(buf1), "https://%s%s", host, buf0); | ||||
|                 if (p_len < 0 || p_len >= sizeof(buf1)) { | ||||
|                     res.status = http_get_status(500); | ||||
|                     print(ERR_STR "Header field 'Location' too long" CLR_STR); | ||||
|                     goto respond; | ||||
|                 } | ||||
|                 http_add_header_field(&res.hdr, "Location", buf1); | ||||
|             } else { | ||||
|                 http_add_header_field(&res.hdr, "Location", buf0); | ||||
|             } | ||||
|             goto respond; | ||||
|         } | ||||
|     } else if (!client->enc) { | ||||
|         res.status = http_get_status(308); | ||||
|         sprintf(buf0, "https://%s%s", host, req.uri); | ||||
|         http_add_header_field(&res.hdr, "Location", buf0); | ||||
|         goto respond; | ||||
|     } | ||||
|  | ||||
|     if (http_get_header_field(&req.hdr, "Transfer-Encoding") != NULL) { | ||||
|         sprintf(err_msg, "This server is unable to process requests with the Transfer-Encoding header field."); | ||||
|         res.status = http_get_status(501); | ||||
|         goto respond; | ||||
|     } | ||||
|  | ||||
|     if (conf->type == CONFIG_TYPE_LOCAL) { | ||||
|         if (uri.filename == NULL && (int) uri.is_static && (int) uri.is_dir && strlen(uri.pathinfo) == 0) { | ||||
|         if (strcmp(req.method, "TRACE") == 0) { | ||||
|             res.status = http_get_status(200); | ||||
|             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", | ||||
|                                       req.method, req.uri, req.version); | ||||
|             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", | ||||
|                                            req.hdr.fields[i][0], req.hdr.fields[i][1]); | ||||
|             } | ||||
|  | ||||
|             goto respond; | ||||
|         } | ||||
|  | ||||
|         if (strncmp(uri.req_path, "/.well-known/", 13) != 0 && strstr(uri.path, "/.") != NULL) { | ||||
|             res.status = http_get_status(403); | ||||
|             sprintf(err_msg, "Parts of this URI are hidden."); | ||||
|             goto respond; | ||||
|         } else if (uri.filename == NULL && (int) uri.is_static && (int) uri.is_dir && strlen(uri.pathinfo) == 0) { | ||||
|             res.status = http_get_status(403); | ||||
|             sprintf(err_msg, "It is not allowed to list the contents of this directory."); | ||||
|             goto respond; | ||||
| @@ -186,13 +241,18 @@ int client_request_handler(sock *client, unsigned long client_num, unsigned int | ||||
|  | ||||
|         if (uri.is_static) { | ||||
|             res.status = http_get_status(200); | ||||
|             http_add_header_field(&res.hdr, "Allow", "GET, HEAD"); | ||||
|             http_add_header_field(&res.hdr, "Accept-Ranges", "bytes"); | ||||
|             if (strcmp(req.method, "GET") != 0 && strcmp(req.method, "HEAD") != 0) { | ||||
|                 res.status = http_get_status(405); | ||||
|                 goto respond; | ||||
|             } | ||||
|  | ||||
|             if (http_get_header_field(&req.hdr, "Content-Length") != NULL) { | ||||
|                 res.status = http_get_status(400); | ||||
|                 sprintf(err_msg, "A GET request must not contain a payload"); | ||||
|                 goto respond; | ||||
|             } | ||||
|  | ||||
|             ret = uri_cache_init(&uri); | ||||
|             if (ret != 0) { | ||||
|                 res.status = http_get_status(500); | ||||
| @@ -215,7 +275,8 @@ int client_request_handler(sock *client, unsigned long client_num, unsigned int | ||||
|             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"); | ||||
|             if ((if_none_match != NULL && strstr(if_none_match, uri.meta->etag) == NULL) || | ||||
|                 (accept_if_modified_since && if_modified_since != NULL && strcmp(if_modified_since, last_modified) == 0)) { | ||||
|                     (accept_if_modified_since && if_modified_since != NULL && | ||||
|                     strcmp(if_modified_since, last_modified) == 0)) { | ||||
|                 res.status = http_get_status(304); | ||||
|                 goto respond; | ||||
|             } | ||||
| @@ -266,7 +327,7 @@ int client_request_handler(sock *client, unsigned long client_num, unsigned int | ||||
|  | ||||
|             char *accept_encoding = http_get_header_field(&req.hdr, "Accept-Encoding"); | ||||
|             if (uri.meta->filename_comp[0] != 0 && accept_encoding != NULL && | ||||
|                 strstr(accept_encoding, "deflate") != NULL) { | ||||
|                     strstr(accept_encoding, "deflate") != NULL) { | ||||
|                 file = fopen(uri.meta->filename_comp, "rb"); | ||||
|                 if (file == NULL) { | ||||
|                     cache_filename_comp_invalid(uri.filename); | ||||
| @@ -293,13 +354,9 @@ int client_request_handler(sock *client, unsigned long client_num, unsigned int | ||||
|                 goto respond; | ||||
|             } | ||||
|  | ||||
|             if (strcmp(req.method, "POST") == 0 || strcmp(req.method, "PUT") == 0) { | ||||
|                 char *client_content_length = http_get_header_field(&req.hdr, "Content-Length"); | ||||
|                 unsigned long client_content_len; | ||||
|                 if (client_content_length == NULL) { | ||||
|                     goto fastcgi_end; | ||||
|                 } | ||||
|                 client_content_len = strtoul(client_content_length, NULL, 10); | ||||
|             char *client_content_length = http_get_header_field(&req.hdr, "Content-Length"); | ||||
|             if (client_content_length != NULL) { | ||||
|                 unsigned long client_content_len = strtoul(client_content_length, NULL, 10); | ||||
|                 ret = fastcgi_receive(&php_fpm, client, client_content_len); | ||||
|                 if (ret != 0) { | ||||
|                     if (ret < 0) { | ||||
| @@ -311,7 +368,6 @@ int client_request_handler(sock *client, unsigned long client_num, unsigned int | ||||
|                     goto respond; | ||||
|                 } | ||||
|             } | ||||
|             fastcgi_end: | ||||
|             fastcgi_close_stdin(&php_fpm); | ||||
|  | ||||
|             ret = fastcgi_header(&php_fpm, &res, err_msg); | ||||
| @@ -338,19 +394,26 @@ int client_request_handler(sock *client, unsigned long client_num, unsigned int | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             char *accept_encoding = http_get_header_field(&req.hdr, "Accept-Encoding"); | ||||
|             if (accept_encoding != NULL && strstr(accept_encoding, "deflate") != NULL) { | ||||
|                 http_add_header_field(&res.hdr, "Content-Encoding", "deflate"); | ||||
|             } | ||||
|  | ||||
|             content_length = -1; | ||||
|             use_fastcgi = 1; | ||||
|  | ||||
|             char *accept_encoding = http_get_header_field(&req.hdr, "Accept-Encoding"); | ||||
|             char *content_type = http_get_header_field(&res.hdr, "Content-Type"); | ||||
|             char *content_encoding = http_get_header_field(&res.hdr, "Content-Encoding"); | ||||
|             if (mime_is_compressible(content_type) && content_encoding == NULL && | ||||
|                     accept_encoding != NULL && strstr(accept_encoding, "deflate") != NULL) { | ||||
|                 http_add_header_field(&res.hdr, "Content-Encoding", "deflate"); | ||||
|                 use_fastcgi = 2; | ||||
|             } | ||||
|  | ||||
|             if (http_get_header_field(&res.hdr, "Content-Length") == NULL) { | ||||
|                 http_add_header_field(&res.hdr, "Transfer-Encoding", "chunked"); | ||||
|             } | ||||
|         } | ||||
|     } else if (conf->type != CONFIG_TYPE_LOCAL) { | ||||
|     } else if (conf->type == CONFIG_TYPE_REVERSE_PROXY) { | ||||
|         print("Reverse proxy for " BLD_STR "%s:%i" CLR_STR, conf->rev_proxy.hostname, conf->rev_proxy.port); | ||||
|         http_remove_header_field(&res.hdr, "Date", HTTP_REMOVE_ALL); | ||||
|         http_remove_header_field(&res.hdr, "Server", HTTP_REMOVE_ALL); | ||||
|         ret = rev_proxy_init(&req, &res, conf, client, &custom_status, err_msg); | ||||
|         use_rev_proxy = ret == 0; | ||||
|     } else { | ||||
| @@ -360,17 +423,27 @@ int client_request_handler(sock *client, unsigned long client_num, unsigned int | ||||
|  | ||||
|     respond: | ||||
|     if (!use_rev_proxy) { | ||||
|         if (conf != NULL && conf->type == CONFIG_TYPE_LOCAL && uri.is_static && res.status->code == 405) { | ||||
|             http_add_header_field(&res.hdr, "Allow", "GET, HEAD, TRACE"); | ||||
|         } | ||||
|         if (http_get_header_field(&res.hdr, "Accept-Ranges") == NULL) { | ||||
|             http_add_header_field(&res.hdr, "Accept-Ranges", "none"); | ||||
|         } | ||||
|         if (!use_fastcgi && !use_rev_proxy && file == NULL && | ||||
|             res.status->code >= 400 && res.status->code < 600) { | ||||
|             http_error_msg *http_msg = http_get_error_msg(res.status->code); | ||||
|             sprintf(msg_pre_buf, http_error_document, res.status->code, res.status->msg, | ||||
|                     http_msg != NULL ? http_msg->err_msg : "", err_msg[0] != 0 ? err_msg : ""); | ||||
|             content_length = sprintf(msg_buf, http_default_document, res.status->code, res.status->msg, | ||||
|                                      msg_pre_buf, res.status->code >= 300 && res.status->code < 400 ? "info" : "error", | ||||
|                                      http_error_icon, "#C00000", host); | ||||
|                 ((res.status->code >= 400 && res.status->code < 600) || err_msg[0] != 0)) { | ||||
|             http_remove_header_field(&res.hdr, "Date", HTTP_REMOVE_ALL); | ||||
|             http_remove_header_field(&res.hdr, "Server", HTTP_REMOVE_ALL); | ||||
|             http_add_header_field(&res.hdr, "Date", http_get_date(buf0, sizeof(buf0))); | ||||
|             http_add_header_field(&res.hdr, "Server", SERVER_STR); | ||||
|  | ||||
|             // TODO list Locations on 3xx Redirects | ||||
|             const http_doc_info *info = http_get_status_info(res.status); | ||||
|             const http_status_msg *http_msg = http_get_error_msg(res.status); | ||||
|  | ||||
|             sprintf(msg_pre_buf, info->doc, res.status->code, res.status->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, | ||||
|                                       res.status->msg, msg_pre_buf, info->mode, info->icon, info->color, host); | ||||
|             http_add_header_field(&res.hdr, "Content-Type", "text/html; charset=UTF-8"); | ||||
|         } | ||||
|         if (content_length >= 0) { | ||||
| @@ -379,12 +452,8 @@ int client_request_handler(sock *client, unsigned long client_num, unsigned int | ||||
|         } else if (http_get_header_field(&res.hdr, "Transfer-Encoding") == NULL) { | ||||
|             server_keep_alive = 0; | ||||
|         } | ||||
|     } else { | ||||
|         http_remove_header_field(&res.hdr, "Server", HTTP_REMOVE_ALL); | ||||
|         http_remove_header_field(&res.hdr, "Date", HTTP_REMOVE_ALL); | ||||
|         http_add_header_field(&res.hdr, "Date", http_get_date(buf0, sizeof(buf0))); | ||||
|         http_add_header_field(&res.hdr, "Server", SERVER_STR); | ||||
|     } | ||||
|  | ||||
|     char *conn = http_get_header_field(&res.hdr, "Connection"); | ||||
|     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); | ||||
| @@ -405,6 +474,8 @@ int client_request_handler(sock *client, unsigned long client_num, unsigned int | ||||
|           res.status->msg, location != NULL ? " -> " : "", location != NULL ? location : "", | ||||
|           format_duration(micros, buf0), CLR_STR); | ||||
|  | ||||
|     // TODO access/error log file | ||||
|  | ||||
|     if (strcmp(req.method, "HEAD") != 0) { | ||||
|         unsigned long snd_len = 0; | ||||
|         unsigned long len; | ||||
| @@ -430,9 +501,7 @@ int client_request_handler(sock *client, unsigned long client_num, unsigned int | ||||
|         } else if (use_fastcgi) { | ||||
|             char *transfer_encoding = http_get_header_field(&res.hdr, "Transfer-Encoding"); | ||||
|             int chunked = transfer_encoding != NULL && strcmp(transfer_encoding, "chunked") == 0; | ||||
|             char *content_encoding = http_get_header_field(&res.hdr, "Content-Encoding"); | ||||
|             int comp = content_encoding != NULL && strcmp(content_encoding, "deflate") == 0; | ||||
|             int flags = (chunked ? FASTCGI_CHUNKED : 0) | (comp ? FASTCGI_COMPRESS : 0); | ||||
|             int flags = (chunked ? FASTCGI_CHUNKED : 0) | ((use_fastcgi == 2) ? FASTCGI_COMPRESS : 0); | ||||
|             fastcgi_send(&php_fpm, client, flags); | ||||
|         } else if (use_rev_proxy) { | ||||
|             char *transfer_encoding = http_get_header_field(&res.hdr, "Transfer-Encoding"); | ||||
|   | ||||
							
								
								
									
										18
									
								
								src/client.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								src/client.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | ||||
| /** | ||||
|  * Necronda Web Server | ||||
|  * Client connection and request handlers (header file) | ||||
|  * src/client.h | ||||
|  * Lorenz Stechauner, 2021-01-17 | ||||
|  */ | ||||
|  | ||||
| #ifndef NECRONDA_SERVER_CLIENT_H | ||||
| #define NECRONDA_SERVER_CLIENT_H | ||||
|  | ||||
| #include <sys/time.h> | ||||
|  | ||||
| extern int server_keep_alive; | ||||
| 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 struct timeval client_timeout; | ||||
|  | ||||
| #endif //NECRONDA_SERVER_CLIENT_H | ||||
| @@ -1,13 +1,26 @@ | ||||
| /**
 | ||||
|  * Necronda Web Server | ||||
|  * File cache implementation | ||||
|  * src/cache.c | ||||
|  * src/lib/cache.c | ||||
|  * Lorenz Stechauner, 2020-12-19 | ||||
|  */ | ||||
| 
 | ||||
| #include <zlib.h> | ||||
| #include "cache.h" | ||||
| #include "uri.h" | ||||
| #include "utils.h" | ||||
| #include "../necronda-server.h" | ||||
| #include <stdio.h> | ||||
| #include <zlib.h> | ||||
| #include <magic.h> | ||||
| #include <sys/ipc.h> | ||||
| #include <sys/shm.h> | ||||
| #include <string.h> | ||||
| #include <errno.h> | ||||
| #include <signal.h> | ||||
| #include <openssl/sha.h> | ||||
| 
 | ||||
| int cache_continue = 1; | ||||
| magic_t magic; | ||||
| cache_entry *cache; | ||||
| 
 | ||||
| int magic_init() { | ||||
|     magic = magic_open(MAGIC_MIME); | ||||
| @@ -69,15 +82,16 @@ int cache_process() { | ||||
|     int compress; | ||||
|     SHA_CTX ctx; | ||||
|     unsigned char hash[SHA_DIGEST_LENGTH]; | ||||
|     int cache_changed = 0; | ||||
|     int p_len; | ||||
|     while (cache_continue) { | ||||
|         for (int i = 0; i < FILE_CACHE_SIZE; i++) { | ||||
|             if (cache[i].filename[0] != 0 && cache[i].meta.etag[0] == 0 && !cache[i].is_updating) { | ||||
|                 cache[i].is_updating = 1; | ||||
|                 fprintf(stdout, "[cache] Hashing file %s\n", cache[i].filename); | ||||
|                 SHA1_Init(&ctx); | ||||
|                 file = fopen(cache[i].filename, "rb"); | ||||
|                 compress = strncmp(cache[i].meta.type, "text/", 5) == 0 || | ||||
|                         (strncmp(cache[i].meta.type, "application/", 12) == 0 && | ||||
|                         strstr(cache[i].meta.type, "+xml") != NULL); | ||||
|                 compress = mime_is_compressible(cache[i].meta.type); | ||||
| 
 | ||||
|                 int level = NECRONDA_ZLIB_LEVEL; | ||||
|                 z_stream strm; | ||||
| @@ -96,11 +110,19 @@ int cache_process() { | ||||
|                         buf[j] = ch; | ||||
|                     } | ||||
|                     buf[strlen(rel_path)] = 0; | ||||
|                     sprintf(filename_comp, "%.*s/.necronda-server/cache/%s.z", cache[i].webroot_len, cache[i].filename, buf); | ||||
|                     p_len = snprintf(filename_comp, sizeof(filename_comp), "%.*s/.necronda-server/cache/%s.z", | ||||
|                                      cache[i].webroot_len, cache[i].filename, buf); | ||||
|                     if (p_len < 0 || p_len >= sizeof(filename_comp)) { | ||||
|                         fprintf(stderr, ERR_STR "Unable to open cached file: " | ||||
|                                                 "File name for compressed file too long" CLR_STR "\n"); | ||||
|                         goto comp_err; | ||||
|                     } | ||||
|                     fprintf(stdout, "[cache] Compressing file %s\n", cache[i].filename); | ||||
|                     comp_file = fopen(filename_comp, "wb"); | ||||
|                     if (comp_file == NULL) { | ||||
|                         fprintf(stderr, ERR_STR "Unable to open cached file: %s" CLR_STR "\n", strerror(errno)); | ||||
|                         comp_err: | ||||
|                         compress = 0; | ||||
|                         fprintf(stderr, ERR_STR "Unable to open cache file: %s" CLR_STR "\n", strerror(errno)); | ||||
|                     } else { | ||||
|                         strm.zalloc = Z_NULL; | ||||
|                         strm.zfree = Z_NULL; | ||||
| @@ -131,6 +153,7 @@ int cache_process() { | ||||
|                 if (compress) { | ||||
|                     deflateEnd(&strm); | ||||
|                     fclose(comp_file); | ||||
|                     fprintf(stdout, "[cache] Finished compressing file %s\n", cache[i].filename); | ||||
|                     strcpy(cache[i].meta.filename_comp, filename_comp); | ||||
|                 } else { | ||||
|                     memset(cache[i].meta.filename_comp, 0, sizeof(cache[i].meta.filename_comp)); | ||||
| @@ -141,14 +164,20 @@ int cache_process() { | ||||
|                     sprintf(cache[i].meta.etag + j * 2, "%02x", hash[j]); | ||||
|                 } | ||||
|                 fclose(file); | ||||
|                 fprintf(stdout, "[cache] Finished hashing file %s\n", cache[i].filename); | ||||
|                 cache[i].is_updating = 0; | ||||
|                 cache_changed = 1; | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
|         cache_file = fopen("/var/necronda-server/cache", "wb"); | ||||
|         fwrite(cache, sizeof(cache_entry), FILE_CACHE_SIZE , cache_file); | ||||
|         fclose(cache_file); | ||||
|         sleep(1); | ||||
|         if (cache_changed) { | ||||
|             cache_changed = 0; | ||||
|             cache_file = fopen("/var/necronda-server/cache", "wb"); | ||||
|             fwrite(cache, sizeof(cache_entry), FILE_CACHE_SIZE, cache_file); | ||||
|             fclose(cache_file); | ||||
|         } else { | ||||
|             sleep(1); | ||||
|         } | ||||
|     } | ||||
|     return 0; | ||||
| } | ||||
| @@ -185,20 +214,18 @@ int cache_init() { | ||||
|     if (pid == 0) { | ||||
|         // child
 | ||||
|         if (cache_process() == 0) { | ||||
|             return 1; | ||||
|             return 0; | ||||
|         } else { | ||||
|             return -6; | ||||
|         } | ||||
|     } else if (pid > 0) { | ||||
|         // parent
 | ||||
|         fprintf(stderr, "Started child process with PID %i as cache-updater\n", pid); | ||||
|         children[0] = pid; | ||||
|         return pid; | ||||
|     } else { | ||||
|         fprintf(stderr, ERR_STR "Unable to create child process: %s" CLR_STR "\n", strerror(errno)); | ||||
|         return -5; | ||||
|     } | ||||
| 
 | ||||
|     return 0; | ||||
| } | ||||
| 
 | ||||
| int cache_unload() { | ||||
| @@ -271,7 +298,7 @@ int cache_filename_comp_invalid(const char *filename) { | ||||
|     int i; | ||||
|     for (i = 0; i < FILE_CACHE_SIZE; i++) { | ||||
|         if (cache[i].filename[0] != 0 && strlen(cache[i].filename) == strlen(filename) && | ||||
|             strcmp(cache[i].filename, filename) == 0) { | ||||
|                 strcmp(cache[i].filename, filename) == 0) { | ||||
|             if (cache[i].is_updating) { | ||||
|                 return 0; | ||||
|             } else { | ||||
| @@ -297,7 +324,7 @@ int uri_cache_init(http_uri *uri) { | ||||
|     int i; | ||||
|     for (i = 0; i < FILE_CACHE_SIZE; i++) { | ||||
|         if (cache[i].filename[0] != 0 && strlen(cache[i].filename) == strlen(uri->filename) && | ||||
|             strcmp(cache[i].filename, uri->filename) == 0) { | ||||
|                 strcmp(cache[i].filename, uri->filename) == 0) { | ||||
|             uri->meta = &cache[i].meta; | ||||
|             if (cache[i].is_updating) { | ||||
|                 return 0; | ||||
| @@ -1,20 +1,15 @@ | ||||
| /**
 | ||||
|  * Necronda Web Server | ||||
|  * File cache implementation (header file) | ||||
|  * src/cache.h | ||||
|  * src/lib/cache.h | ||||
|  * Lorenz Stechauner, 2020-12-19 | ||||
|  */ | ||||
| 
 | ||||
| #ifndef NECRONDA_SERVER_CACHE_H | ||||
| #define NECRONDA_SERVER_CACHE_H | ||||
| 
 | ||||
| #include <magic.h> | ||||
| #include <sys/ipc.h> | ||||
| #include <sys/shm.h> | ||||
| #include "uri.h" | ||||
| 
 | ||||
| magic_t magic; | ||||
| 
 | ||||
| typedef struct { | ||||
|     char filename[256]; | ||||
|     unsigned char webroot_len; | ||||
| @@ -22,9 +17,9 @@ typedef struct { | ||||
|     meta_data meta; | ||||
| } cache_entry; | ||||
| 
 | ||||
| cache_entry *cache; | ||||
| extern cache_entry *cache; | ||||
| 
 | ||||
| int cache_continue = 1; | ||||
| extern int cache_continue; | ||||
| 
 | ||||
| int magic_init(); | ||||
| 
 | ||||
| @@ -1,14 +1,22 @@ | ||||
| /**
 | ||||
|  * Necronda Web Server | ||||
|  * Configuration file loader | ||||
|  * src/config.c | ||||
|  * src/lib/config.c | ||||
|  * Lorenz Stechauner, 2021-01-05 | ||||
|  */ | ||||
| 
 | ||||
| #include "config.h" | ||||
| #include "uri.h" | ||||
| #include "../necronda-server.h" | ||||
| #include <stdio.h> | ||||
| #include <sys/ipc.h> | ||||
| #include <sys/shm.h> | ||||
| #include <string.h> | ||||
| #include <errno.h> | ||||
| #include <malloc.h> | ||||
| #include <stdlib.h> | ||||
| 
 | ||||
| host_config *config; | ||||
| char cert_file[256], key_file[256], geoip_dir[256], dns_server[256]; | ||||
| 
 | ||||
| int config_init() { | ||||
|     int shm_id = shmget(SHM_KEY_CONFIG, MAX_HOST_CONFIG * sizeof(host_config), IPC_CREAT | IPC_EXCL | 0640); | ||||
| @@ -71,8 +79,7 @@ int config_load(const char *filename) { | ||||
|     int i = 0; | ||||
|     int mode = 0; | ||||
|     char *ptr = NULL; | ||||
|     char host[256], *source, *target; | ||||
|     host[0] = 0; | ||||
|     char *source, *target; | ||||
|     while ((ptr = strtok(ptr == NULL ? conf :  NULL, "\n")) != NULL) { | ||||
|         char *comment = strchr(ptr, '#'); | ||||
|         if (comment != NULL) comment[0] = 0; | ||||
| @@ -177,9 +184,17 @@ int config_load(const char *filename) { | ||||
|             tmp_config[i - 1].rev_proxy.port = (unsigned short) strtoul(source, NULL, 10); | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     free(conf); | ||||
| 
 | ||||
|     for (int j = 0; j < i; j++) { | ||||
|         if (tmp_config[j].type == CONFIG_TYPE_LOCAL) { | ||||
|             char *webroot = tmp_config[j].local.webroot; | ||||
|             if (webroot[strlen(webroot) - 1] == '/') { | ||||
|                 webroot[strlen(webroot) - 1] = 0; | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     int shm_id = shmget(SHM_KEY_CONFIG, 0, 0); | ||||
|     if (shm_id < 0) { | ||||
|         fprintf(stderr, ERR_STR "Unable to get shared memory id: %s" CLR_STR "\n", strerror(errno)); | ||||
| @@ -1,19 +1,19 @@ | ||||
| /**
 | ||||
|  * Necronda Web Server | ||||
|  * Configuration file loader (header file) | ||||
|  * src/config.h | ||||
|  * src/lib/config.h | ||||
|  * Lorenz Stechauner, 2021-01-05 | ||||
|  */ | ||||
| 
 | ||||
| 
 | ||||
| #ifndef NECRONDA_SERVER_CONFIG_H | ||||
| #define NECRONDA_SERVER_CONFIG_H | ||||
| 
 | ||||
| #include "uri.h" | ||||
| 
 | ||||
| #define CONFIG_TYPE_UNSET 0 | ||||
| #define CONFIG_TYPE_LOCAL 1 | ||||
| #define CONFIG_TYPE_REVERSE_PROXY 2 | ||||
| 
 | ||||
| 
 | ||||
| typedef struct { | ||||
|     int type; | ||||
|     char name[256]; | ||||
| @@ -30,10 +30,8 @@ typedef struct { | ||||
|     }; | ||||
| } host_config; | ||||
| 
 | ||||
| 
 | ||||
| host_config *config; | ||||
| char cert_file[256], key_file[256], geoip_dir[256], dns_server[256]; | ||||
| 
 | ||||
| extern host_config *config; | ||||
| extern char cert_file[256], key_file[256], geoip_dir[256], dns_server[256]; | ||||
| 
 | ||||
| int config_init(); | ||||
| 
 | ||||
| @@ -1,15 +1,19 @@ | ||||
| /**
 | ||||
|  * Necronda Web Server | ||||
|  * FastCGI interface implementation | ||||
|  * src/fastcgi.c | ||||
|  * src/lib/fastcgi.c | ||||
|  * Lorenz Stechauner, 2020-12-26 | ||||
|  */ | ||||
| 
 | ||||
| #include "fastcgi.h" | ||||
| #include "necronda-server.h" | ||||
| 
 | ||||
| #include "utils.h" | ||||
| #include "../client.h" | ||||
| #include "../necronda-server.h" | ||||
| #include <sys/un.h> | ||||
| 
 | ||||
| #include <zlib.h> | ||||
| #include <sys/socket.h> | ||||
| #include <errno.h> | ||||
| #include <string.h> | ||||
| 
 | ||||
| char *fastcgi_add_param(char *buf, const char *key, const char *value) { | ||||
|     char *ptr = buf; | ||||
| @@ -133,7 +137,7 @@ int fastcgi_init(fastcgi_conn *conn, unsigned int client_num, unsigned int req_n | ||||
|     if (uri->pathinfo != NULL && strlen(uri->pathinfo) > 0) { | ||||
|         sprintf(buf0, "/%s", uri->pathinfo); | ||||
|     } else { | ||||
|         sprintf(buf0, ""); | ||||
|         buf0[0] = 0; | ||||
|     } | ||||
|     param_ptr = fastcgi_add_param(param_ptr, "PATH_INFO", buf0); | ||||
| 
 | ||||
| @@ -205,17 +209,21 @@ int fastcgi_close_stdin(fastcgi_conn *conn) { | ||||
|     return 0; | ||||
| } | ||||
| 
 | ||||
| int fastcgi_php_error(char *msg, int msg_len, char *err_msg) { | ||||
| int fastcgi_php_error(const char *msg, int msg_len, char *err_msg) { | ||||
|     char *msg_str = malloc(msg_len + 1); | ||||
|     char *ptr0 = msg_str; | ||||
|     strncpy(msg_str, msg, msg_len); | ||||
|     char *ptr1 = NULL; | ||||
|     int len; | ||||
|     int err = 0; | ||||
|     // FIXME *msg is part of a stream, handle fragmented lines
 | ||||
|     while (1) { | ||||
|         int msg_type = 0; | ||||
|         int msg_pre_len = 0; | ||||
|         ptr1 = strstr(ptr0, "PHP message: "); | ||||
|         if (ptr1 == NULL) { | ||||
|             len = (int) (msg_len - (ptr0 - msg_str)); | ||||
|             if (ptr0 == msg_str) msg_type = 2; | ||||
|         } else { | ||||
|             len = (int) (ptr1 - ptr0); | ||||
|         } | ||||
| @@ -223,8 +231,6 @@ int fastcgi_php_error(char *msg, int msg_len, char *err_msg) { | ||||
|             goto next; | ||||
|         } | ||||
| 
 | ||||
|         int msg_type = 0; | ||||
|         int msg_pre_len = 0; | ||||
|         if (len >= 14 && strncmp(ptr0, "PHP Warning:  ", 14) == 0) { | ||||
|             msg_type = 1; | ||||
|             msg_pre_len = 14; | ||||
| @@ -248,7 +254,7 @@ int fastcgi_php_error(char *msg, int msg_len, char *err_msg) { | ||||
|             if (ptr3 != NULL && (ptr3 - ptr2) < len2) { | ||||
|                 len2 = (int) (ptr3 - ptr2); | ||||
|             } | ||||
|             print("%s%.*s%s", msg_type == 1 ? WRN_STR : msg_type == 2 ? ERR_STR: "", len2, ptr2, msg_type != 0 ? CLR_STR : ""); | ||||
|             print("%s%.*s%s", msg_type == 1 ? WRN_STR : msg_type == 2 ? ERR_STR : "", len2, ptr2, msg_type != 0 ? CLR_STR : ""); | ||||
|             if (msg_type == 2 && ptr2 == ptr0) { | ||||
|                 sprintf(err_msg, "%.*s", len2, ptr2); | ||||
|                 err = 1; | ||||
| @@ -265,6 +271,7 @@ int fastcgi_php_error(char *msg, int msg_len, char *err_msg) { | ||||
|         } | ||||
|         ptr0 = ptr1 + 13; | ||||
|     } | ||||
|     free(msg_str); | ||||
|     return err; | ||||
| } | ||||
| 
 | ||||
| @@ -457,7 +464,7 @@ int fastcgi_send(fastcgi_conn *conn, sock *client, int flags) { | ||||
| 
 | ||||
|             return 0; | ||||
|         } else if (header.type == FCGI_STDERR) { | ||||
|             print(ERR_STR "%.*s" CLR_STR, content_len, content); | ||||
|             fastcgi_php_error(content, content_len, buf0); | ||||
|         } else if (header.type == FCGI_STDOUT) { | ||||
|             out: | ||||
|             if (flags & FASTCGI_COMPRESS) { | ||||
							
								
								
									
										41
									
								
								src/lib/fastcgi.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										41
									
								
								src/lib/fastcgi.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,41 @@ | ||||
| /** | ||||
|  * Necronda Web Server | ||||
|  * FastCGI interface implementation (header file) | ||||
|  * src/lib/fastcgi.h | ||||
|  * Lorenz Stechauner, 2020-12-26 | ||||
|  */ | ||||
|  | ||||
| #ifndef NECRONDA_SERVER_FASTCGI_H | ||||
| #define NECRONDA_SERVER_FASTCGI_H | ||||
|  | ||||
| #include "include/fastcgi.h" | ||||
| #include "http.h" | ||||
| #include "uri.h" | ||||
|  | ||||
| #define FASTCGI_CHUNKED 1 | ||||
| #define FASTCGI_COMPRESS 2 | ||||
|  | ||||
| typedef struct { | ||||
|     int socket; | ||||
|     unsigned short req_id; | ||||
|     char *out_buf; | ||||
|     unsigned short out_len; | ||||
|     unsigned short out_off; | ||||
| } fastcgi_conn; | ||||
|  | ||||
| char *fastcgi_add_param(char *buf, const char *key, const char *value); | ||||
|  | ||||
| int fastcgi_init(fastcgi_conn *conn, unsigned int client_num, unsigned int req_num, const sock *client, | ||||
|                  const http_req *req, const http_uri *uri); | ||||
|  | ||||
| int fastcgi_close_stdin(fastcgi_conn *conn); | ||||
|  | ||||
| int fastcgi_php_error(const char *msg, int msg_len, char *err_msg); | ||||
|  | ||||
| int fastcgi_header(fastcgi_conn *conn, http_res *res, char *err_msg); | ||||
|  | ||||
| int fastcgi_send(fastcgi_conn *conn, sock *client, int flags); | ||||
|  | ||||
| int fastcgi_receive(fastcgi_conn *conn, sock *client, unsigned long len); | ||||
|  | ||||
| #endif //NECRONDA_SERVER_FASTCGI_H | ||||
| @@ -1,13 +1,14 @@ | ||||
| /**
 | ||||
|  * Necronda Web Server | ||||
|  * HTTP implementation | ||||
|  * src/net/http.c | ||||
|  * src/lib/http.c | ||||
|  * Lorenz Stechauner, 2020-12-09 | ||||
|  */ | ||||
| 
 | ||||
| #include "http.h" | ||||
| #include "utils.h" | ||||
| 
 | ||||
| #include "../necronda-server.h" | ||||
| #include <string.h> | ||||
| 
 | ||||
| void http_to_camel_case(char *str, int mode) { | ||||
|     char last = '-'; | ||||
| @@ -50,9 +51,9 @@ int http_parse_header_field(http_hdr *hdr, const char *buf, const char *end_ptr) | ||||
|     } | ||||
| 
 | ||||
|     long len = pos1 - buf; | ||||
|     hdr->fields[hdr->field_num][0] = malloc(len + 1); | ||||
|     sprintf(hdr->fields[hdr->field_num][0], "%.*s", (int) len, buf); | ||||
|     http_to_camel_case(hdr->fields[hdr->field_num][0], HTTP_CAMEL); | ||||
|     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++; | ||||
|     pos2 = (char *) end_ptr - 1; | ||||
| @@ -61,11 +62,11 @@ int http_parse_header_field(http_hdr *hdr, const char *buf, const char *end_ptr) | ||||
|     len = pos2 - pos1 + 1; | ||||
| 
 | ||||
|     if (len <= 0) { | ||||
|         hdr->fields[hdr->field_num][1] = malloc(1); | ||||
|         hdr->fields[hdr->field_num][1][0] = 0; | ||||
|         hdr->fields[(int) hdr->field_num][1] = malloc(1); | ||||
|         hdr->fields[(int) hdr->field_num][1][0] = 0; | ||||
|     } else { | ||||
|         hdr->fields[hdr->field_num][1] = malloc(len + 1); | ||||
|         sprintf(hdr->fields[hdr->field_num][1], "%.*s", (int) len, pos1); | ||||
|         hdr->fields[(int) hdr->field_num][1] = malloc(len + 1); | ||||
|         sprintf(hdr->fields[(int) hdr->field_num][1], "%.*s", (int) len, pos1); | ||||
|     } | ||||
|     hdr->field_num++; | ||||
|     return 0; | ||||
| @@ -185,8 +186,8 @@ void http_add_header_field(http_hdr *hdr, const char *field_name, const char *fi | ||||
|     strcpy(_field_name, field_name); | ||||
|     strcpy(_field_value, field_value); | ||||
|     http_to_camel_case(_field_name, HTTP_PRESERVE); | ||||
|     hdr->fields[hdr->field_num][0] = _field_name; | ||||
|     hdr->fields[hdr->field_num][1] = _field_value; | ||||
|     hdr->fields[(int) hdr->field_num][0] = _field_name; | ||||
|     hdr->fields[(int) hdr->field_num][1] = _field_value; | ||||
|     hdr->field_num++; | ||||
| } | ||||
| 
 | ||||
| @@ -245,8 +246,8 @@ int http_send_request(sock *server, http_req *req) { | ||||
|     return 0; | ||||
| } | ||||
| 
 | ||||
| http_status *http_get_status(unsigned short status_code) { | ||||
|     for (int i = 0; i < sizeof(http_statuses) / sizeof(http_status); i++) { | ||||
| const http_status *http_get_status(unsigned short status_code) { | ||||
|     for (int i = 0; i < http_statuses_size / sizeof(http_status); i++) { | ||||
|         if (http_statuses[i].code == status_code) { | ||||
|             return &http_statuses[i]; | ||||
|         } | ||||
| @@ -254,20 +255,21 @@ http_status *http_get_status(unsigned short status_code) { | ||||
|     return NULL; | ||||
| } | ||||
| 
 | ||||
| http_error_msg *http_get_error_msg(unsigned short status_code) { | ||||
|     for (int i = 0; i < sizeof(http_error_messages) / sizeof(http_error_msg); i++) { | ||||
|         if (http_error_messages[i].code == status_code) { | ||||
|             return &http_error_messages[i]; | ||||
| const http_status_msg *http_get_error_msg(const http_status *status) { | ||||
|     unsigned short code = status->code; | ||||
|     for (int i = 0; i < http_status_messages_size / sizeof(http_status_msg); i++) { | ||||
|         if (http_status_messages[i].code == code) { | ||||
|             return &http_status_messages[i]; | ||||
|         } | ||||
|     } | ||||
|     return NULL; | ||||
| } | ||||
| 
 | ||||
| const char *http_get_status_color(http_status *status) { | ||||
| const char *http_get_status_color(const http_status *status) { | ||||
|     unsigned short code = status->code; | ||||
|     if (code >= 100 && code < 200) { | ||||
|         return HTTP_1XX_STR; | ||||
|     } else if (code >= 200 && code < 300 || code == 304) { | ||||
|     } else if ((code >= 200 && code < 300) || code == 304) { | ||||
|         return HTTP_2XX_STR; | ||||
|     } else if (code >= 300 && code < 400) { | ||||
|         return HTTP_3XX_STR; | ||||
| @@ -290,3 +292,23 @@ char *http_get_date(char *buf, size_t size) { | ||||
|     time(&rawtime); | ||||
|     return http_format_date(rawtime, buf, size); | ||||
| } | ||||
| 
 | ||||
| const http_doc_info *http_get_status_info(const http_status *status) { | ||||
|     unsigned short code = status->code; | ||||
|     static http_doc_info info[] = { | ||||
|             {"info", HTTP_COLOR_INFO, http_info_icon, http_info_document}, | ||||
|             {"success", HTTP_COLOR_SUCCESS, http_success_icon, http_success_document}, | ||||
|             {"warning", HTTP_COLOR_WARNING, http_warning_icon, http_warning_document}, | ||||
|             {"error", HTTP_COLOR_ERROR, http_error_icon, http_error_document} | ||||
|     }; | ||||
|     if (code >= 100 && code < 200) { | ||||
|         return &info[0]; | ||||
|     } else if ((code >= 200 && code < 300) || code == 304) { | ||||
|         return &info[1]; | ||||
|     } else if (code >= 300 && code < 400) { | ||||
|         return &info[2]; | ||||
|     } else if (code >= 400 && code < 600) { | ||||
|         return &info[3]; | ||||
|     } | ||||
|     return NULL; | ||||
| } | ||||
							
								
								
									
										111
									
								
								src/lib/http.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										111
									
								
								src/lib/http.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,111 @@ | ||||
| /** | ||||
|  * Necronda Web Server | ||||
|  * HTTP implementation (header file) | ||||
|  * src/lib/http.h | ||||
|  * Lorenz Stechauner, 2020-12-09 | ||||
|  */ | ||||
|  | ||||
| #ifndef NECRONDA_SERVER_HTTP_H | ||||
| #define NECRONDA_SERVER_HTTP_H | ||||
|  | ||||
| #include "sock.h" | ||||
|  | ||||
| #define HTTP_PRESERVE 0 | ||||
| #define HTTP_LOWER 1 | ||||
| #define HTTP_CAMEL 2 | ||||
|  | ||||
| #define HTTP_REMOVE_ONE 0 | ||||
| #define HTTP_REMOVE_ALL 1 | ||||
| #define HTTP_REMOVE_LAST 2 | ||||
|  | ||||
| #define HTTP_COLOR_SUCCESS "#008000" | ||||
| #define HTTP_COLOR_INFO "#606060" | ||||
| #define HTTP_COLOR_WARNING "#E0C000" | ||||
| #define HTTP_COLOR_ERROR "#C00000" | ||||
|  | ||||
| typedef struct { | ||||
|     unsigned short code; | ||||
|     char type[16]; | ||||
|     char msg[32]; | ||||
| } http_status; | ||||
|  | ||||
| typedef struct { | ||||
|     unsigned short code; | ||||
|     const char *msg; | ||||
| } http_status_msg; | ||||
|  | ||||
| typedef struct { | ||||
|     char mode[8]; | ||||
|     char color[8]; | ||||
|     const char *icon; | ||||
|     const char *doc; | ||||
| } http_doc_info; | ||||
|  | ||||
| typedef struct { | ||||
|     char field_num; | ||||
|     char *fields[64][2]; | ||||
| } http_hdr; | ||||
|  | ||||
| typedef struct { | ||||
|     char method[16]; | ||||
|     char *uri; | ||||
|     char version[3]; | ||||
|     http_hdr hdr; | ||||
| } http_req; | ||||
|  | ||||
| typedef struct { | ||||
|     const http_status *status; | ||||
|     char version[3]; | ||||
|     http_hdr hdr; | ||||
| } http_res; | ||||
|  | ||||
| extern const http_status http_statuses[]; | ||||
| extern const http_status_msg http_status_messages[]; | ||||
| extern const int http_statuses_size; | ||||
| extern const int http_status_messages_size; | ||||
|  | ||||
| extern const char http_default_document[]; | ||||
| extern const char http_error_document[]; | ||||
| extern const char http_error_icon[]; | ||||
| extern const char http_warning_document[]; | ||||
| extern const char http_warning_icon[]; | ||||
| extern const char http_success_document[]; | ||||
| extern const char http_success_icon[]; | ||||
| extern const char http_info_document[]; | ||||
| extern const char http_info_icon[]; | ||||
|  | ||||
| void http_to_camel_case(char *str, int mode); | ||||
|  | ||||
| void http_free_hdr(http_hdr *hdr); | ||||
|  | ||||
| void http_free_req(http_req *req); | ||||
|  | ||||
| void http_free_res(http_res *res); | ||||
|  | ||||
| int http_receive_request(sock *client, http_req *req); | ||||
|  | ||||
| int http_parse_header_field(http_hdr *hdr, const char *buf, const char *end_ptr) ; | ||||
|  | ||||
| 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); | ||||
|  | ||||
| void http_remove_header_field(http_hdr *hdr, const char *field_name, int mode); | ||||
|  | ||||
| int http_send_response(sock *client, http_res *res); | ||||
|  | ||||
| int http_send_request(sock *server, http_req *req); | ||||
|  | ||||
| const http_status *http_get_status(unsigned short status_code); | ||||
|  | ||||
| const http_status_msg *http_get_error_msg(const http_status *status); | ||||
|  | ||||
| const char *http_get_status_color(const http_status *status); | ||||
|  | ||||
| char *http_format_date(time_t time, char *buf, size_t size); | ||||
|  | ||||
| char *http_get_date(char *buf, size_t size); | ||||
|  | ||||
| const http_doc_info *http_get_status_info(const http_status *status); | ||||
|  | ||||
| #endif //NECRONDA_SERVER_HTTP_H | ||||
| @@ -1,51 +1,14 @@ | ||||
| /**
 | ||||
|  * Necronda Web Server | ||||
|  * HTTP implementation (header file) | ||||
|  * src/net/http.h | ||||
|  * Lorenz Stechauner, 2020-12-09 | ||||
|  * HTTP static implementation | ||||
|  * src/lib/http_static.c | ||||
|  * Lorenz Stechauner, 2021-05-03 | ||||
|  */ | ||||
| 
 | ||||
| #ifndef NECRONDA_SERVER_HTTP_H | ||||
| #define NECRONDA_SERVER_HTTP_H | ||||
| #include "http.h" | ||||
| #include "../necronda-server.h" | ||||
| 
 | ||||
| #define HTTP_PRESERVE 0 | ||||
| #define HTTP_LOWER 1 | ||||
| #define HTTP_CAMEL 2 | ||||
| 
 | ||||
| #define HTTP_REMOVE_ONE 0 | ||||
| #define HTTP_REMOVE_ALL 1 | ||||
| #define HTTP_REMOVE_LAST 2 | ||||
| 
 | ||||
| typedef struct { | ||||
|     unsigned short code; | ||||
|     char type[16]; | ||||
|     char msg[32]; | ||||
| } http_status; | ||||
| 
 | ||||
| typedef struct { | ||||
|     unsigned short code; | ||||
|     char *err_msg; | ||||
| } http_error_msg; | ||||
| 
 | ||||
| typedef struct { | ||||
|     char field_num; | ||||
|     char *fields[64][2]; | ||||
| } http_hdr; | ||||
| 
 | ||||
| typedef struct { | ||||
|     char method[16]; | ||||
|     char *uri; | ||||
|     char version[3]; | ||||
|     http_hdr hdr; | ||||
| } http_req; | ||||
| 
 | ||||
| typedef struct { | ||||
|     http_status *status; | ||||
|     char version[3]; | ||||
|     http_hdr hdr; | ||||
| } http_res; | ||||
| 
 | ||||
| http_status http_statuses[] = { | ||||
| const http_status http_statuses[] = { | ||||
|         {100, "Informational", "Continue"}, | ||||
|         {101, "Informational", "Switching Protocols"}, | ||||
| 
 | ||||
| @@ -61,7 +24,7 @@ http_status http_statuses[] = { | ||||
|         {301, "Redirection",   "Moved Permanently"}, | ||||
|         {302, "Redirection",   "Found"}, | ||||
|         {303, "Redirection",   "See Other"}, | ||||
|         {304, "Redirection",   "Not Modified"}, | ||||
|         {304, "Success",       "Not Modified"}, | ||||
|         {305, "Redirection",   "Use Proxy"}, | ||||
|         {307, "Redirection",   "Temporary Redirect"}, | ||||
|         {308, "Redirection",   "Permanent Redirect"}, | ||||
| @@ -93,7 +56,27 @@ http_status http_statuses[] = { | ||||
|         {505, "Server Error",  "HTTP Version Not Supported"}, | ||||
| }; | ||||
| 
 | ||||
| http_error_msg http_error_messages[] = { | ||||
| const http_status_msg http_status_messages[] = { | ||||
|         {100, "The client SHOULD continue with its request."}, | ||||
|         {101, "The server understands and is willing to comply with the clients request, via the Upgrade message header field, for a change in the application protocol being used on this connection."}, | ||||
| 
 | ||||
|         {200, "The request has succeeded."}, | ||||
|         {201, "The request has been fulfilled and resulted in a new resource being created."}, | ||||
|         {202, "The request has been accepted for processing, but the processing has not been completed."}, | ||||
|         {203, "The returned meta information in the entity-header is not the definitive set as available from the origin server, but is gathered from a local or a third-party copy."}, | ||||
|         {204, "The server has fulfilled the request but does not need to return an entity-body, and might want to return updated meta information."}, | ||||
|         {205, "The server has fulfilled the request and the user agent SHOULD reset the document view which caused the request to be sent."}, | ||||
|         {206, "The server has fulfilled the partial GET request for the resource."}, | ||||
| 
 | ||||
|         {300, "The requested resource corresponds to any one of a set of representations, each with its own specific location, and agent-driven negotiation information is being provided so that the user (or user agent) can select a preferred representation and redirect its request to that location."}, | ||||
|         {301, "The requested resource has been assigned a new permanent URI and any future references to this resource SHOULD use one of the returned URIs."}, | ||||
|         {302, "The requested resource resides temporarily under a different URI."}, | ||||
|         {303, "The response to the request can be found under a different URI and SHOULD be retrieved using a GET method on that resource."}, | ||||
|         {304, "The request has been fulfilled and the requested resource has not been modified."}, | ||||
|         {305, "The requested resource MUST be accessed through the proxy given by the Location field."}, | ||||
|         {307, "The requested resource resides temporarily under a different URI."}, | ||||
|         {308, "The requested resource has been assigned a new permanent URI and any future references to this resource ought to use one of the enclosed URIs."}, | ||||
| 
 | ||||
|         {400, "The request could not be understood by the server due to malformed syntax."}, | ||||
|         {401, "The request requires user authentication."}, | ||||
|         {403, "The server understood the request, but is refusing to fulfill it."}, | ||||
| @@ -109,7 +92,7 @@ http_error_msg http_error_messages[] = { | ||||
|         {413, "The server is refusing to process a request because the request entity is larger than the server is willing or able to process."}, | ||||
|         {414, "The server is refusing to service the request because the Request-URI is longer than the server is willing to interpret."}, | ||||
|         {415, "The server is refusing to service the request because the entity of the request is in a format not supported by the requested resource for the requested method."}, | ||||
|         {416, "None of the ranges in the request's Range header field overlap the current extent of the selected resource or that the set of ranges requested has been rejected due to invalid ranges or an excessive request of small or overlapping ranges."}, | ||||
|         {416, "None of the ranges in the requests Range header field overlap the current extent of the selected resource or that the set of ranges requested has been rejected due to invalid ranges or an excessive request of small or overlapping ranges."}, | ||||
|         {417, "The expectation given in an Expect request-header field could not be met by this server, or, if the server is a proxy, the server has unambiguous evidence that the request could not be met by the next-hop server."}, | ||||
| 
 | ||||
|         {500, "The server encountered an unexpected condition which prevented it from fulfilling the request."}, | ||||
| @@ -120,7 +103,7 @@ http_error_msg http_error_messages[] = { | ||||
|         {505, "The server does not support, or refuses to support, the HTTP protocol version that was used in the request message."} | ||||
| }; | ||||
| 
 | ||||
| const char *http_default_document = | ||||
| const char http_default_document[] = | ||||
|         "<!DOCTYPE html>\n" | ||||
|         "<html lang=\"en\">\n" | ||||
|         "<head>\n" | ||||
| @@ -130,12 +113,13 @@ const char *http_default_document = | ||||
|         "\t<meta name=\"color-scheme\" content=\"light dark\"/>\n" | ||||
|         "\t<meta name=\"apple-mobile-web-app-status-bar-style\" content=\"black-translucent\"/>\n" | ||||
|         "\t<meta name=\"viewport\" content=\"width=device-width,initial-scale=1.0\"/>\n" | ||||
|         "\t<link rel=\"shortcut icon\" type=\"image/x-icon\" href=\"/favicon.ico\"/>\n" | ||||
|         "%5$s" | ||||
|         "\t<style>\n" | ||||
|         "\t\thtml{font-family:\"Arial\",sans-serif;--error:#C00000;--info:#E0C000;--color:var(--%4$s);}\n" | ||||
|         "\t\tbody{background-color:#F0F0F0;margin:0.5em;}\n" | ||||
|         "\t\thtml{font-family:\"Arial\",sans-serif;--error:" HTTP_COLOR_ERROR ";--warning:" HTTP_COLOR_WARNING ";--success:" HTTP_COLOR_SUCCESS ";--info:" HTTP_COLOR_INFO ";--color:var(--%4$s);}\n" | ||||
|         "\t\tbody{background-color:#F0F0F0;margin:0;}\n" | ||||
|         "\t\tmain{max-width:650px;margin:2em auto;}\n" | ||||
|         "\t\tsection{margin:1em;background-color:#FFFFFF;border: 1px solid var(--color);border-radius:4px;padding:1em 2em;}\n" | ||||
|         "\t\tsection{margin:1em;background-color:#FFFFFF;border: 1px solid var(--color);border-radius:4px;padding:1em;}\n" | ||||
|         "\t\th1,h2,h3,h4,h5,h6,h7{text-align:center;color:var(--color);font-weight:normal;}\n" | ||||
|         "\t\th1{font-size:3em;margin:0.125em 0 0.125em 0;}\n" | ||||
|         "\t\th2{font-size:1.5em;margin:0.25em 0 1em 0;}\n" | ||||
| @@ -153,55 +137,62 @@ const char *http_default_document = | ||||
|         "\t<main>\n" | ||||
|         "\t\t<section>\n" | ||||
|         "%3$s" | ||||
|         "\t\t\t<div class=\"footer\"><a href=\"https://%7$s/\">%7$s</a> - Necronda web server " NECRONDA_VERSION "</div>\n" | ||||
|         "\t\t\t<div class=\"footer\"><a href=\"https://%7$s/\">%7$s</a> - Necronda web server " NECRONDA_VERSION "</div>\n" | ||||
|         "\t\t</section>\n" | ||||
|         "\t</main>\n" | ||||
|         "</body>\n" | ||||
|         "</html>\n"; | ||||
| 
 | ||||
| const char *http_error_document = | ||||
| const char http_error_document[] = | ||||
|         "\t\t\t<h1>%1$i</h1>\n" | ||||
|         "\t\t\t<h2>%2$s :(</h2>\n" | ||||
|         "\t\t\t<p>%3$s</p>\n" | ||||
|         "\t\t\t<p>%4$s</p>\n"; | ||||
| 
 | ||||
| const char *http_error_icon = | ||||
|         "\t<link rel=\"shortcut icon\" type=\"image/svg+xml\" sizes=\"any\" href=\"data:image/svg+xml;base64," | ||||
| const char http_error_icon[] = | ||||
|         "\t<link rel=\"alternate icon\" type=\"image/svg+xml\" sizes=\"any\" href=\"data:image/svg+xml;base64," | ||||
|         "PHN2ZyB3aWR0aD0iMTYiIGhlaWdodD0iMTYiIHhtbG5zPSJodHRwOi8vd3d3LnczLm9yZy8yMDAw" | ||||
|         "L3N2ZyI+PHRleHQgeD0iNCIgeT0iMTIiIGZpbGw9IiNDMDAwMDAiIHN0eWxlPSJmb250LWZhbWls" | ||||
|         "eTonQXJpYWwnLHNhbnMtc2VyaWYiPjooPC90ZXh0Pjwvc3ZnPgo=\"/>\n"; | ||||
| 
 | ||||
| 
 | ||||
| void http_to_camel_case(char *str, int mode); | ||||
| const char http_warning_document[] = | ||||
|         "\t\t\t<h1>%1$i</h1>\n" | ||||
|         "\t\t\t<h2>%2$s :o</h2>\n" | ||||
|         "\t\t\t<p>%3$s</p>\n" | ||||
|         "\t\t\t<p>%4$s</p>\n"; | ||||
| 
 | ||||
| void http_free_hdr(http_hdr *hdr); | ||||
| const char http_warning_icon[] = | ||||
|         "\t<link rel=\"alternate icon\" type=\"image/svg+xml\" sizes=\"any\" href=\"data:image/svg+xml;base64," | ||||
|         "PHN2ZyB3aWR0aD0iMTYiIGhlaWdodD0iMTYiIHhtbG5zPSJodHRwOi8vd3d3LnczLm9yZy8yMDAw" | ||||
|         "L3N2ZyI+PHRleHQgeD0iNCIgeT0iMTIiIGZpbGw9IiNFMEMwMDAiIHN0eWxlPSJmb250LWZhbWls" | ||||
|         "eTonQXJpYWwnLHNhbnMtc2VyaWYiPjpvPC90ZXh0Pjwvc3ZnPgo=\"/>\n"; | ||||
| 
 | ||||
| void http_free_req(http_req *req); | ||||
| 
 | ||||
| void http_free_res(http_res *res); | ||||
| const char http_success_document[] = | ||||
|         "\t\t\t<h1>%1$i</h1>\n" | ||||
|         "\t\t\t<h2>%2$s :)</h2>\n" | ||||
|         "\t\t\t<p>%3$s</p>\n" | ||||
|         "\t\t\t<p>%4$s</p>\n"; | ||||
| 
 | ||||
| int http_receive_request(sock *client, http_req *req); | ||||
| const char http_success_icon[] = | ||||
|         "\t<link rel=\"alternate icon\" type=\"image/svg+xml\" sizes=\"any\" href=\"data:image/svg+xml;base64," | ||||
|         "PHN2ZyB3aWR0aD0iMTYiIGhlaWdodD0iMTYiIHhtbG5zPSJodHRwOi8vd3d3LnczLm9yZy8yMDAw" | ||||
|         "L3N2ZyI+PHRleHQgeD0iNCIgeT0iMTIiIGZpbGw9IiMwMDgwMDAiIHN0eWxlPSJmb250LWZhbWls" | ||||
|         "eTonQXJpYWwnLHNhbnMtc2VyaWYiPjopPC90ZXh0Pjwvc3ZnPgo=\"/>\n"; | ||||
| 
 | ||||
| int http_parse_header_field(http_hdr *hdr, const char *buf, const char *end_ptr) ; | ||||
| 
 | ||||
| char *http_get_header_field(const http_hdr *hdr, const char *field_name); | ||||
| const char http_info_document[] = | ||||
|         "\t\t\t<h1>%1$i</h1>\n" | ||||
|         "\t\t\t<h2>%2$s :)</h2>\n" | ||||
|         "\t\t\t<p>%3$s</p>\n" | ||||
|         "\t\t\t<p>%4$s</p>\n"; | ||||
| 
 | ||||
| void http_add_header_field(http_hdr *hdr, const char *field_name, const char *field_value); | ||||
| const char http_info_icon[] = | ||||
|         "\t<link rel=\"alternate icon\" type=\"image/svg+xml\" sizes=\"any\" href=\"data:image/svg+xml;base64," | ||||
|         "PHN2ZyB3aWR0aD0iMTYiIGhlaWdodD0iMTYiIHhtbG5zPSJodHRwOi8vd3d3LnczLm9yZy8yMDAw" | ||||
|         "L3N2ZyI+PHRleHQgeD0iNCIgeT0iMTIiIGZpbGw9IiM2MDYwNjAiIHN0eWxlPSJmb250LWZhbWls" | ||||
|         "eTonQXJpYWwnLHNhbnMtc2VyaWYiPjopPC90ZXh0Pjwvc3ZnPgo=\"/>\n"; | ||||
| 
 | ||||
| void http_remove_header_field(http_hdr *hdr, const char *field_name, int mode); | ||||
| 
 | ||||
| int http_send_response(sock *client, http_res *res); | ||||
| 
 | ||||
| int http_send_request(sock *server, http_req *req); | ||||
| 
 | ||||
| http_status *http_get_status(unsigned short status_code); | ||||
| 
 | ||||
| http_error_msg *http_get_error_msg(unsigned short status_code); | ||||
| 
 | ||||
| const char *http_get_status_color(http_status *status); | ||||
| 
 | ||||
| char *http_format_date(time_t time, char *buf, size_t size); | ||||
| 
 | ||||
| char *http_get_date(char *buf, size_t size); | ||||
| 
 | ||||
| #endif //NECRONDA_SERVER_HTTP_H
 | ||||
| const int http_statuses_size = sizeof(http_statuses); | ||||
| const int http_status_messages_size = sizeof(http_status_messages); | ||||
| @@ -1,37 +1,12 @@ | ||||
| /**
 | ||||
|  * Necronda Web Server | ||||
|  * FastCGI interface implementation (header file) | ||||
|  * src/fastcgi.h | ||||
|  * Lorenz Stechauner, 2020-12-26 | ||||
|  * FastCGI header file | ||||
|  * src/lib/include/fastcgi.h | ||||
|  * Lorenz Stechauner, 2021-05-03 | ||||
|  */ | ||||
| 
 | ||||
| #ifndef NECRONDA_SERVER_FASTCGI_H | ||||
| #define NECRONDA_SERVER_FASTCGI_H | ||||
| 
 | ||||
| #define FASTCGI_CHUNKED 1 | ||||
| #define FASTCGI_COMPRESS 2 | ||||
| 
 | ||||
| typedef struct { | ||||
|     int socket; | ||||
|     unsigned short req_id; | ||||
|     char *out_buf; | ||||
|     unsigned short out_len; | ||||
|     unsigned short out_off; | ||||
| } fastcgi_conn; | ||||
| 
 | ||||
| char *fastcgi_add_param(char *buf, const char *key, const char *value); | ||||
| 
 | ||||
| int fastcgi_init(fastcgi_conn *conn, unsigned int client_num, unsigned int req_num, const sock *client, | ||||
|                  const http_req *req, const http_uri *uri); | ||||
| 
 | ||||
| int fastcgi_close_stdin(fastcgi_conn *conn); | ||||
| 
 | ||||
| int fastcgi_header(fastcgi_conn *conn, http_res *res, char *err_msg); | ||||
| 
 | ||||
| int fastcgi_send(fastcgi_conn *conn, sock *client, int flags); | ||||
| 
 | ||||
| int fastcgi_receive(fastcgi_conn *conn, sock *client, unsigned long len); | ||||
| 
 | ||||
| #ifndef NECRONDA_SERVER_EXTERN_FASTCGI_H | ||||
| #define NECRONDA_SERVER_EXTERN_FASTCGI_H | ||||
| 
 | ||||
| /*
 | ||||
|  * Listening socket file number | ||||
| @@ -144,4 +119,4 @@ typedef struct { | ||||
|     FCGI_UnknownTypeBody body; | ||||
| } FCGI_UnknownTypeRecord; | ||||
| 
 | ||||
| #endif //NECRONDA_SERVER_FASTCGI_H
 | ||||
| #endif //NECRONDA_SERVER_EXTERN_FASTCGI_H
 | ||||
| @@ -1,19 +1,161 @@ | ||||
| /**
 | ||||
|  * Necronda Web Server | ||||
|  * Reverse proxy | ||||
|  * src/rev_proxy.c | ||||
|  * src/lib/rev_proxy.c | ||||
|  * Lorenz Stechauner, 2021-01-07 | ||||
|  */ | ||||
| 
 | ||||
| #include "rev_proxy.h" | ||||
| #include "utils.h" | ||||
| #include "../client.h" | ||||
| #include "../necronda-server.h" | ||||
| #include <openssl/ssl.h> | ||||
| #include <string.h> | ||||
| #include <errno.h> | ||||
| #include <openssl/err.h> | ||||
| #include <arpa/inet.h> | ||||
| #include <sys/time.h> | ||||
| 
 | ||||
| sock rev_proxy; | ||||
| char *rev_proxy_host = NULL; | ||||
| struct timeval server_timeout = {.tv_sec = SERVER_TIMEOUT, .tv_usec = 0}; | ||||
| 
 | ||||
| int rev_proxy_preload() { | ||||
|     rev_proxy.buf = NULL; | ||||
|     rev_proxy.buf_len = 0; | ||||
|     rev_proxy.buf_off = 0; | ||||
|     rev_proxy.ctx = SSL_CTX_new(TLS_client_method()); | ||||
|     return 0; | ||||
| } | ||||
| 
 | ||||
| int rev_proxy_request_header(http_req *req, int enc) { | ||||
|     char buf1[256]; | ||||
|     char buf2[256]; | ||||
|     int p_len; | ||||
|     http_remove_header_field(&req->hdr, "Connection", HTTP_REMOVE_ALL); | ||||
|     http_add_header_field(&req->hdr, "Connection", "keep-alive"); | ||||
| 
 | ||||
|     char *via = http_get_header_field(&req->hdr, "Via"); | ||||
|     sprintf(buf1, "HTTP/%s %s", req->version, DEFAULT_HOST); | ||||
|     if (via == NULL) { | ||||
|         http_add_header_field(&req->hdr, "Via", buf1); | ||||
|     } else { | ||||
|         p_len = snprintf(buf2, sizeof(buf2), "%s, %s", via, buf1); | ||||
|         if (p_len < 0 || p_len >= sizeof(buf2)) { | ||||
|             print(ERR_STR "Header field 'Via' too long" CLR_STR); | ||||
|             return -1; | ||||
|         } | ||||
|         http_remove_header_field(&req->hdr, "Via", HTTP_REMOVE_ALL); | ||||
|         http_add_header_field(&req->hdr, "Via", buf2); | ||||
|     } | ||||
| 
 | ||||
|     char *host = http_get_header_field(&req->hdr, "Host"); | ||||
|     char *forwarded = http_get_header_field(&req->hdr, "Forwarded"); | ||||
|     int client_ipv6 = strchr(client_addr_str, ':') != NULL; | ||||
|     int server_ipv6 =  strchr(server_addr_str, ':') != NULL; | ||||
| 
 | ||||
|     p_len = snprintf(buf1, sizeof(buf1), "by=%s%s%s;for=%s%s%s;host=%s;proto=%s", | ||||
|                      server_ipv6 ? "\"[" : "", server_addr_str, server_ipv6 ? "]\"" : "", | ||||
|                      client_ipv6 ? "\"[" : "", client_addr_str, client_ipv6 ? "]\"" : "", | ||||
|                      host, enc ? "https" : "http"); | ||||
|     if (p_len < 0 || p_len >= sizeof(buf1)) { | ||||
|         print(ERR_STR "Appended part of header field 'Forwarded' too long" CLR_STR); | ||||
|         return -1; | ||||
|     } | ||||
|     if (forwarded == NULL) { | ||||
|         http_add_header_field(&req->hdr, "Forwarded", buf1); | ||||
|     } else { | ||||
|         p_len = snprintf(buf2, sizeof(buf2), "%s, %s", forwarded, buf1); | ||||
|         if (p_len < 0 || p_len >= sizeof(buf2)) { | ||||
|             print(ERR_STR "Header field 'Forwarded' too long" CLR_STR); | ||||
|             return -1; | ||||
|         } | ||||
|         http_remove_header_field(&req->hdr, "Forwarded", HTTP_REMOVE_ALL); | ||||
|         http_add_header_field(&req->hdr, "Forwarded", buf2); | ||||
|     } | ||||
| 
 | ||||
|     char *xff = http_get_header_field(&req->hdr, "X-Forwarded-For"); | ||||
|     if (xff == NULL) { | ||||
|         http_add_header_field(&req->hdr, "X-Forwarded-For", client_addr_str); | ||||
|     } else { | ||||
|         sprintf(buf1, "%s, %s", xff, client_addr_str); | ||||
|         http_remove_header_field(&req->hdr, "X-Forwarded-For", HTTP_REMOVE_ALL); | ||||
|         http_add_header_field(&req->hdr, "X-Forwarded-For", buf1); | ||||
|     } | ||||
| 
 | ||||
|     char *xfh = http_get_header_field(&req->hdr, "X-Forwarded-Host"); | ||||
|     if (xfh == NULL) { | ||||
|         if (forwarded == NULL) { | ||||
|             http_add_header_field(&req->hdr, "X-Forwarded-Host", host); | ||||
|         } else { | ||||
|             char *ptr = strchr(forwarded, ','); | ||||
|             unsigned long len; | ||||
|             if (ptr != NULL) len = ptr - forwarded; | ||||
|             else len = strlen(forwarded); | ||||
|             ptr = strstr(forwarded, "host="); | ||||
|             if ((ptr - forwarded) < len) { | ||||
|                 char *end = strchr(ptr, ';'); | ||||
|                 if (end == NULL) len -= (ptr - forwarded); | ||||
|                 else len = (end - ptr); | ||||
|                 len -= 5; | ||||
|                 sprintf(buf1, "%.*s", (int) len, ptr + 5); | ||||
|                 http_add_header_field(&req->hdr, "X-Forwarded-Host", buf1); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     char *xfp = http_get_header_field(&req->hdr, "X-Forwarded-Proto"); | ||||
|     if (xfp == NULL) { | ||||
|         if (forwarded == NULL) { | ||||
|             http_add_header_field(&req->hdr, "X-Forwarded-Proto", enc ? "https" : "http"); | ||||
|         } else { | ||||
|             char *ptr = strchr(forwarded, ','); | ||||
|             unsigned long len; | ||||
|             if (ptr != NULL) len = ptr - forwarded; | ||||
|             else len = strlen(forwarded); | ||||
|             ptr = strstr(forwarded, "proto="); | ||||
|             if ((ptr - forwarded) < len) { | ||||
|                 char *end = strchr(ptr, ';'); | ||||
|                 if (end == NULL) len -= (ptr - forwarded); | ||||
|                 else len = (end - ptr); | ||||
|                 len -= 6; | ||||
|                 sprintf(buf1, "%.*s", (int) len, ptr + 6); | ||||
|                 http_add_header_field(&req->hdr, "X-Forwarded-Proto", buf1); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     return 0; | ||||
| } | ||||
| 
 | ||||
| int rev_proxy_response_header(http_req *req, http_res *res) { | ||||
|     char buf1[256]; | ||||
|     char buf2[256]; | ||||
|     int p_len; | ||||
| 
 | ||||
|     char *via = http_get_header_field(&res->hdr, "Via"); | ||||
|     p_len = snprintf(buf1, sizeof(buf1), "HTTP/%s %s", req->version, DEFAULT_HOST); | ||||
|     if (p_len < 0 || p_len >= sizeof(buf1)) { | ||||
|         print(ERR_STR "Appended part of header field 'Via' too long" CLR_STR); | ||||
|         return -1; | ||||
|     } | ||||
|     if (via == NULL) { | ||||
|         http_add_header_field(&res->hdr, "Via", buf1); | ||||
|     } else { | ||||
|         p_len = snprintf(buf2, sizeof(buf2), "%s, %s", via, buf1); | ||||
|         if (p_len < 0 || p_len >= sizeof(buf2)) { | ||||
|             print(ERR_STR "Header field 'Via' too long" CLR_STR); | ||||
|             return -1; | ||||
|         } | ||||
|         http_remove_header_field(&res->hdr, "Via", HTTP_REMOVE_ALL); | ||||
|         http_add_header_field(&res->hdr, "Via", buf2); | ||||
|     } | ||||
| 
 | ||||
|     return 0; | ||||
| } | ||||
| 
 | ||||
| int rev_proxy_init(http_req *req, http_res *res, host_config *conf, sock *client, http_status *custom_status, | ||||
|                    char * err_msg) { | ||||
|                    char *err_msg) { | ||||
|     char buffer[CHUNK_SIZE]; | ||||
|     long ret; | ||||
|     int tries = 0; | ||||
| @@ -97,10 +239,11 @@ int rev_proxy_init(http_req *req, http_res *res, host_config *conf, sock *client | ||||
|     print(BLUE_STR "Established new connection with " BLD_STR "[%s]:%i" CLR_STR, buffer, conf->rev_proxy.port); | ||||
| 
 | ||||
|     rev_proxy: | ||||
|     http_remove_header_field(&req->hdr, "Connection", HTTP_REMOVE_ALL); | ||||
|     http_add_header_field(&req->hdr, "Connection", "keep-alive"); | ||||
|     http_remove_header_field(&req->hdr, "X-Forwarded-For", HTTP_REMOVE_ALL); | ||||
|     http_add_header_field(&req->hdr, "X-Forwarded-For", client_addr_str); | ||||
|     ret = rev_proxy_request_header(req, (int) client->enc); | ||||
|     if (ret != 0) { | ||||
|         res->status = http_get_status(500); | ||||
|         return -1; | ||||
|     } | ||||
| 
 | ||||
|     ret = http_send_request(&rev_proxy, req); | ||||
|     if (ret < 0) { | ||||
| @@ -222,6 +365,12 @@ int rev_proxy_init(http_req *req, http_res *res, host_config *conf, sock *client | ||||
|     } | ||||
|     sock_recv(&rev_proxy, buffer, header_len, 0); | ||||
| 
 | ||||
|     ret = rev_proxy_response_header(req, res); | ||||
|     if (ret != 0) { | ||||
|         res->status = http_get_status(500); | ||||
|         return -1; | ||||
|     } | ||||
| 
 | ||||
|     return 0; | ||||
| 
 | ||||
|     proxy_err: | ||||
							
								
								
									
										27
									
								
								src/lib/rev_proxy.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								src/lib/rev_proxy.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,27 @@ | ||||
| /** | ||||
|  * Necronda Web Server | ||||
|  * Reverse proxy (header file) | ||||
|  * src/lib/rev_proxy.h | ||||
|  * Lorenz Stechauner, 2021-01-07 | ||||
|  */ | ||||
|  | ||||
| #ifndef NECRONDA_SERVER_REV_PROXY_H | ||||
| #define NECRONDA_SERVER_REV_PROXY_H | ||||
|  | ||||
| #include "http.h" | ||||
| #include "config.h" | ||||
|  | ||||
| extern sock rev_proxy; | ||||
|  | ||||
| int rev_proxy_preload(); | ||||
|  | ||||
| int rev_proxy_request_header(http_req *req, int enc); | ||||
|  | ||||
| int rev_proxy_response_header(http_req *req, http_res *res); | ||||
|  | ||||
| int rev_proxy_init(http_req *req, http_res *res, host_config *conf, sock *client, http_status *custom_status, | ||||
|                    char *err_msg); | ||||
|  | ||||
| int rev_proxy_send(sock *client, int chunked, unsigned long len_to_send); | ||||
|  | ||||
| #endif //NECRONDA_SERVER_REV_PROXY_H | ||||
| @@ -1,11 +1,16 @@ | ||||
| /**
 | ||||
|  * Necronda Web Server | ||||
|  * Basic TCP and TLS socket | ||||
|  * src/sock.c | ||||
|  * src/lib/sock.c | ||||
|  * Lorenz Stechauner, 2021-01-07 | ||||
|  */ | ||||
| 
 | ||||
| #include "sock.h" | ||||
| #include <openssl/err.h> | ||||
| #include <openssl/ssl.h> | ||||
| #include <string.h> | ||||
| #include <sys/socket.h> | ||||
| #include <unistd.h> | ||||
| 
 | ||||
| const char *sock_strerror(sock *s) { | ||||
|     if (s->_last_ret == 0) { | ||||
| @@ -81,7 +86,7 @@ long sock_splice(sock *dst, sock *src, void *buf, unsigned long buf_len, unsigne | ||||
|         next_len = (buf_len < (len - send_len)) ? buf_len : (len - send_len); | ||||
|         ret = sock_recv(src, buf, next_len, 0); | ||||
|         if (ret < 0) return -2; | ||||
|         if (ret != next_len) return -3; | ||||
|         next_len = ret; | ||||
|         ret = sock_send(dst, buf, next_len, send_len + next_len < len ? MSG_MORE : 0); | ||||
|         if (ret < 0) return -1; | ||||
|         if (ret != next_len) return -3; | ||||
| @@ -1,13 +1,15 @@ | ||||
| /**
 | ||||
|  * Necronda Web Server | ||||
|  * Basic TCP and TLS socket (header file) | ||||
|  * src/sock.h | ||||
|  * src/lib/sock.h | ||||
|  * Lorenz Stechauner, 2021-01-07 | ||||
|  */ | ||||
| 
 | ||||
| #ifndef NECRONDA_SERVER_SOCK_H | ||||
| #define NECRONDA_SERVER_SOCK_H | ||||
| 
 | ||||
| #include <openssl/crypto.h> | ||||
| 
 | ||||
| typedef struct { | ||||
|     unsigned int enc:1; | ||||
|     int socket; | ||||
| @@ -1,12 +1,14 @@ | ||||
| /**
 | ||||
|  * Necronda Web Server | ||||
|  * URI and path handlers | ||||
|  * src/uri.c | ||||
|  * src/lib/uri.c | ||||
|  * Lorenz Stechauner, 2020-12-13 | ||||
|  */ | ||||
| 
 | ||||
| #include "uri.h" | ||||
| 
 | ||||
| #include "utils.h" | ||||
| #include <stdlib.h> | ||||
| #include <string.h> | ||||
| 
 | ||||
| int path_is_directory(const char *path) { | ||||
|     struct stat statbuf; | ||||
| @@ -28,6 +30,7 @@ int uri_init(http_uri *uri, const char *webroot, const char *uri_str, int dir_mo | ||||
|     char buf1[1024]; | ||||
|     char buf2[1024]; | ||||
|     char buf3[1024]; | ||||
|     int p_len; | ||||
|     uri->webroot = NULL; | ||||
|     uri->req_path = NULL; | ||||
|     uri->path = NULL; | ||||
| @@ -50,12 +53,12 @@ int uri_init(http_uri *uri, const char *webroot, const char *uri_str, int dir_mo | ||||
|     } else { | ||||
|         query[0] = 0; | ||||
|         query++; | ||||
|         ssize_t size = strlen(query) + 1; | ||||
|         long size = (long) strlen(query) + 1; | ||||
|         uri->query = malloc(size); | ||||
|         strcpy(uri->query, query); | ||||
|     } | ||||
| 
 | ||||
|     ssize_t size = strlen(uri_str) + 1; | ||||
|     long size = (long) strlen(uri_str) + 1; | ||||
|     uri->req_path = malloc(size); | ||||
|     url_decode(uri_str, uri->req_path, &size); | ||||
|     if (query != NULL) { | ||||
| @@ -65,7 +68,7 @@ int uri_init(http_uri *uri, const char *webroot, const char *uri_str, int dir_mo | ||||
|         return 2; | ||||
|     } | ||||
| 
 | ||||
|     size = strlen(uri->req_path) + 1; | ||||
|     size = (long) strlen(uri->req_path) + 1; | ||||
|     uri->path = malloc(size); | ||||
|     uri->pathinfo = malloc(size); | ||||
| 
 | ||||
| @@ -89,10 +92,17 @@ int uri_init(http_uri *uri, const char *webroot, const char *uri_str, int dir_mo | ||||
|     } else { | ||||
|         strcpy(uri->pathinfo, ""); | ||||
|     } | ||||
| 
 | ||||
|     if (!path_exists(uri->webroot)) { | ||||
|         return 3; | ||||
|     } | ||||
| 
 | ||||
|     while (1) { | ||||
|         sprintf(buf0, "%s%s", uri->webroot, uri->path); | ||||
|         sprintf(buf1, "%s.php", buf0); | ||||
|         sprintf(buf2, "%s.html", buf0); | ||||
|         p_len = snprintf(buf1, sizeof(buf1), "%s.php", buf0); | ||||
|         if (p_len < 0 || p_len >= sizeof(buf1)) return -1; | ||||
|         p_len = snprintf(buf2, sizeof(buf2), "%s.html", buf0); | ||||
|         if (p_len < 0 || p_len >= sizeof(buf2)) return -1; | ||||
| 
 | ||||
|         if (strlen(uri->path) <= 1 || path_exists(buf0) || path_is_file(buf1) || path_is_file(buf2)) { | ||||
|             break; | ||||
| @@ -101,7 +111,7 @@ int uri_init(http_uri *uri, const char *webroot, const char *uri_str, int dir_mo | ||||
|         char *ptr; | ||||
|         parent_dir: | ||||
|         ptr = strrchr(uri->path, '/'); | ||||
|         size = strlen(ptr); | ||||
|         size = (long) strlen(ptr); | ||||
|         sprintf(buf3, "%.*s%s", (int) size, ptr, uri->pathinfo); | ||||
|         strcpy(uri->pathinfo, buf3); | ||||
|         ptr[0] = 0; | ||||
| @@ -114,7 +124,7 @@ int uri_init(http_uri *uri, const char *webroot, const char *uri_str, int dir_mo | ||||
|     if (path_is_file(buf0)) { | ||||
|         uri->filename = malloc(strlen(buf0) + 1); | ||||
|         strcpy(uri->filename, buf0); | ||||
|         ssize_t len = strlen(uri->path); | ||||
|         long len = (long) strlen(uri->path); | ||||
|         if (strcmp(uri->path + len - 5, ".html") == 0) { | ||||
|             uri->path[len - 5] = 0; | ||||
|         } else if (strcmp(uri->path + len - 4, ".php") == 0) { | ||||
| @@ -1,7 +1,7 @@ | ||||
| /**
 | ||||
|  * Necronda Web Server | ||||
|  * URI and path handlers (header file) | ||||
|  * src/uri.h | ||||
|  * src/lib/uri.h | ||||
|  * Lorenz Stechauner, 2020-12-13 | ||||
|  */ | ||||
| 
 | ||||
| @@ -1,11 +1,15 @@ | ||||
| /**
 | ||||
|  * Necronda Web Server | ||||
|  * Utilities | ||||
|  * src/utils.c | ||||
|  * src/lib/utils.c | ||||
|  * Lorenz Stechauner, 2020-12-03 | ||||
|  */ | ||||
| 
 | ||||
| #include "utils.h" | ||||
| #include <string.h> | ||||
| #include <stdlib.h> | ||||
| 
 | ||||
| char *log_prefix; | ||||
| 
 | ||||
| char *format_duration(unsigned long micros, char *buf) { | ||||
|     if (micros < 10000) { | ||||
| @@ -22,7 +26,7 @@ char *format_duration(unsigned long micros, char *buf) { | ||||
|     return buf; | ||||
| } | ||||
| 
 | ||||
| int url_encode(const char *str, char *enc, ssize_t *size) { | ||||
| int url_encode_component(const char *str, char *enc, ssize_t *size) { | ||||
|     char *ptr = enc; | ||||
|     char ch; | ||||
|     memset(enc, 0, *size); | ||||
| @@ -32,8 +36,8 @@ int url_encode(const char *str, char *enc, ssize_t *size) { | ||||
|         } | ||||
|         ch = str[i]; | ||||
|         if (ch == ':' || ch == '/' || ch == '?' || ch == '#' || ch == '[' || ch == ']' || ch == '@' || ch == '!' || | ||||
|             ch == '$' || ch == '&' || ch == '\'' || ch == '(' || ch == ')' || ch == '*' || ch == '+' || ch == ',' || | ||||
|             ch == ';' || ch == '=' || ch < ' ' || ch > '~') { | ||||
|                 ch == '$' || ch == '&' || ch == '\'' || ch == '(' || ch == ')' || ch == '*' || ch == '+' || ch == ',' || | ||||
|                 ch == ';' || ch == '=' || ch < ' ' || ch > '~') { | ||||
|             if ((ptr - enc + 2) >= *size) { | ||||
|                 return -1; | ||||
|             } | ||||
| @@ -49,7 +53,7 @@ int url_encode(const char *str, char *enc, ssize_t *size) { | ||||
|     return 0; | ||||
| } | ||||
| 
 | ||||
| int encode_url(const char *str, char *enc, ssize_t *size) { | ||||
| int url_encode(const char *str, char *enc, ssize_t *size) { | ||||
|     char *ptr = enc; | ||||
|     unsigned char ch; | ||||
|     memset(enc, 0, *size); | ||||
| @@ -88,6 +92,9 @@ int url_decode(const char *str, char *dec, ssize_t *size) { | ||||
|             buf[2] = 0; | ||||
|             ch = (char) strtol(buf, NULL, 16); | ||||
|             i += 2; | ||||
|         } else if (ch == '?') { | ||||
|             strcpy(ptr, str + i); | ||||
|             break; | ||||
|         } | ||||
|         ptr[0] = ch; | ||||
|     } | ||||
| @@ -95,6 +102,20 @@ int url_decode(const char *str, char *dec, ssize_t *size) { | ||||
|     return 0; | ||||
| } | ||||
| 
 | ||||
| int mime_is_compressible(const char *type) { | ||||
|     return | ||||
|         strncmp(type, "text/", 5) == 0 || | ||||
|         strncmp(type, "message/", 7) == 0 || | ||||
|         strstr(type, "+xml") != NULL || | ||||
|         strcmp(type, "application/javascript") == 0 || | ||||
|         strcmp(type, "application/json") == 0 || | ||||
|         strcmp(type, "application/xml") == 0 || | ||||
|         strcmp(type, "application/x-www-form-urlencoded") == 0 || | ||||
|         strcmp(type, "application/x-tex") == 0 || | ||||
|         strcmp(type, "application/x-httpd-php") == 0 || | ||||
|         strcmp(type, "application/x-latex") == 0; | ||||
| } | ||||
| 
 | ||||
| MMDB_entry_data_list_s *mmdb_json(MMDB_entry_data_list_s *list, char *str, long *str_off, long str_len) { | ||||
|     switch (list->entry_data.type) { | ||||
|         case MMDB_DATA_TYPE_MAP: | ||||
| @@ -116,7 +137,7 @@ MMDB_entry_data_list_s *mmdb_json(MMDB_entry_data_list_s *list, char *str, long | ||||
|             *str_off += sprintf(str + *str_off, "%lu", list->entry_data.uint64); | ||||
|             break; | ||||
|         case MMDB_DATA_TYPE_UINT128: | ||||
|             *str_off += sprintf(str + *str_off, "%llu", list->entry_data.uint128); | ||||
|             *str_off += sprintf(str + *str_off, "%llu", (unsigned long long) list->entry_data.uint128); | ||||
|             break; | ||||
|         case MMDB_DATA_TYPE_INT32: | ||||
|             *str_off += sprintf(str + *str_off, "%i", list->entry_data.uint32); | ||||
| @@ -1,14 +1,16 @@ | ||||
| /**
 | ||||
|  * Necronda Web Server | ||||
|  * Utilities (header file) | ||||
|  * src/utils.h | ||||
|  * src/lib/utils.h | ||||
|  * Lorenz Stechauner, 2020-12-03 | ||||
|  */ | ||||
| 
 | ||||
| #ifndef NECRONDA_SERVER_UTILS_H | ||||
| #define NECRONDA_SERVER_UTILS_H | ||||
| 
 | ||||
| char *log_prefix; | ||||
| #include <maxminddb.h> | ||||
| 
 | ||||
| extern char *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) | ||||
| @@ -22,10 +24,14 @@ char *log_prefix; | ||||
| 
 | ||||
| char *format_duration(unsigned long micros, char *buf); | ||||
| 
 | ||||
| int url_encode(const char *str, char *enc, ssize_t *size); | ||||
| int url_encode_component(const char *str, char *enc, ssize_t *size); | ||||
| 
 | ||||
| int encode_url(const char *str, char *enc, ssize_t *size); | ||||
| int url_encode(const char *str, char *enc, ssize_t *size); | ||||
| 
 | ||||
| int url_decode(const char *str, char *dec, ssize_t *size); | ||||
| 
 | ||||
| int mime_is_compressible(const char *type); | ||||
| 
 | ||||
| MMDB_entry_data_list_s *mmdb_json(MMDB_entry_data_list_s *list, char *str, long *str_off, long str_len); | ||||
| 
 | ||||
| #endif //NECRONDA_SERVER_UTILS_H
 | ||||
| @@ -8,21 +8,35 @@ | ||||
| #define _POSIX_C_SOURCE 199309L | ||||
|  | ||||
| #include "necronda-server.h" | ||||
|  | ||||
| #include "config.c" | ||||
| #include "utils.c" | ||||
| #include "uri.c" | ||||
| #include "cache.c" | ||||
| #include "sock.c" | ||||
| #include "http.c" | ||||
| #include "rev_proxy.c" | ||||
| #include "client.c" | ||||
| #include "fastcgi.c" | ||||
|  | ||||
| #include "lib/cache.h" | ||||
| #include "lib/config.h" | ||||
| #include "lib/sock.h" | ||||
| #include "lib/rev_proxy.h" | ||||
|  | ||||
| #include <stdio.h> | ||||
| #include <sys/socket.h> | ||||
| #include <signal.h> | ||||
| #include <unistd.h> | ||||
| #include <sys/select.h> | ||||
| #include <string.h> | ||||
| #include <errno.h> | ||||
| #include <arpa/inet.h> | ||||
| #include <wait.h> | ||||
| #include <sys/types.h> | ||||
| #include <openssl/err.h> | ||||
| #include <openssl/pem.h> | ||||
| #include <openssl/ssl.h> | ||||
| #include <openssl/conf.h> | ||||
| #include <maxminddb.h> | ||||
| #include <dirent.h> | ||||
|  | ||||
| int active = 1; | ||||
| const char *config_file; | ||||
|  | ||||
| int sockets[NUM_SOCKETS]; | ||||
| pid_t children[MAX_CHILDREN]; | ||||
| MMDB_s mmdbs[MAX_MMDB]; | ||||
|  | ||||
| void openssl_init() { | ||||
|     SSL_library_init(); | ||||
| @@ -273,10 +287,7 @@ int main(int argc, const char *argv[]) { | ||||
|     SSL_CTX_set_cipher_list(client.ctx, "HIGH:!aNULL:!kRSA:!PSK:!SRP:!MD5:!RC4"); | ||||
|     SSL_CTX_set_ecdh_auto(client.ctx, 1); | ||||
|  | ||||
|     rev_proxy.buf = NULL; | ||||
|     rev_proxy.buf_len = 0; | ||||
|     rev_proxy.buf_off = 0; | ||||
|     rev_proxy.ctx = SSL_CTX_new(TLS_client_method()); | ||||
|     rev_proxy_preload(); | ||||
|  | ||||
|     if (SSL_CTX_use_certificate_chain_file(client.ctx, cert_file) != 1) { | ||||
|         fprintf(stderr, ERR_STR "Unable to load certificate chain file: %s: %s" CLR_STR "\n", | ||||
| @@ -312,6 +323,8 @@ int main(int argc, const char *argv[]) { | ||||
|         config_unload(); | ||||
|         return 1; | ||||
|     } else if (ret != 0) { | ||||
|         children[0] = ret;  // pid | ||||
|     } else { | ||||
|         return 0; | ||||
|     } | ||||
|  | ||||
|   | ||||
| @@ -8,26 +8,8 @@ | ||||
| #ifndef NECRONDA_SERVER_NECRONDA_SERVER_H | ||||
| #define NECRONDA_SERVER_NECRONDA_SERVER_H | ||||
|  | ||||
| #include <stdio.h> | ||||
| #include <sys/socket.h> | ||||
| #include <signal.h> | ||||
| #include <unistd.h> | ||||
| #include <sys/select.h> | ||||
| #include <string.h> | ||||
| #include <errno.h> | ||||
| #include <arpa/inet.h> | ||||
| #include <wait.h> | ||||
| #include <sys/types.h> | ||||
| #include <stdio.h> | ||||
| #include <openssl/err.h> | ||||
| #include <openssl/pem.h> | ||||
| #include <openssl/ssl.h> | ||||
| #include <openssl/conf.h> | ||||
| #include <openssl/engine.h> | ||||
| #include <openssl/dh.h> | ||||
| #include <maxminddb.h> | ||||
| #include <dirent.h> | ||||
|  | ||||
|  | ||||
| #define NUM_SOCKETS 2 | ||||
| #define MAX_CHILDREN 1024 | ||||
| @@ -60,7 +42,7 @@ | ||||
| #define HTTP_4XX_STR "\x1B[1;31m" | ||||
| #define HTTP_5XX_STR "\x1B[1;31m" | ||||
|  | ||||
| #define NECRONDA_VERSION "4.2" | ||||
| #define NECRONDA_VERSION "4.3" | ||||
| #define SERVER_STR "Necronda/" NECRONDA_VERSION | ||||
| #define NECRONDA_ZLIB_LEVEL 9 | ||||
|  | ||||
| @@ -77,10 +59,8 @@ | ||||
| #define DEFAULT_CONFIG_FILE "/etc/necronda-server/necronda-server.conf" | ||||
| #endif | ||||
|  | ||||
| int sockets[NUM_SOCKETS]; | ||||
| pid_t children[MAX_CHILDREN]; | ||||
| MMDB_s mmdbs[MAX_MMDB]; | ||||
|  | ||||
| char *client_addr_str, *client_addr_str_ptr, *server_addr_str, *server_addr_str_ptr, *client_host_str; | ||||
| extern int sockets[NUM_SOCKETS]; | ||||
| extern pid_t children[MAX_CHILDREN]; | ||||
| extern MMDB_s mmdbs[MAX_MMDB]; | ||||
|  | ||||
| #endif //NECRONDA_SERVER_NECRONDA_SERVER_H | ||||
|   | ||||
| @@ -1,11 +0,0 @@ | ||||
| /** | ||||
|  * Necronda Web Server | ||||
|  * Reverse proxy (header file) | ||||
|  * src/rev_proxy.h | ||||
|  * Lorenz Stechauner, 2021-01-07 | ||||
|  */ | ||||
|  | ||||
| #ifndef NECRONDA_SERVER_REV_PROXY_H | ||||
| #define NECRONDA_SERVER_REV_PROXY_H | ||||
|  | ||||
| #endif //NECRONDA_SERVER_REV_PROXY_H | ||||
		Reference in New Issue
	
	Block a user