Compare commits
	
		
			33 Commits
		
	
	
		
			ee8aedce91
			...
			stable-4
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 081af44eba | |||
| 782c7440b0 | |||
| 88346fe722 | |||
| 0f75aeea7a | |||
| 933aac0f09 | |||
| 7f1299feb4 | |||
| 8435048150 | |||
| 206ae3264d | |||
| 5e050512ad | |||
| db053121f2 | |||
| 89a9d4b9d6 | |||
| bc7c3591a2 | |||
| 1859c432c8 | |||
| 33ec943e8b | |||
| b30f9fa56d | |||
| 90e324cf87 | |||
| e7e1e7b18f | |||
| 63b1ca5d6a | |||
| 54313551fc | |||
| 92779e5dba | |||
| 100eb1597d | |||
| f4c3345445 | |||
| 893316ebfa | |||
| 6a511732af | |||
| bb895c5bca | |||
| a2f4bf57e0 | |||
| 635271ec50 | |||
| 2b4569aabe | |||
| f361fce561 | |||
| c92742275a | |||
| 041e4d43a7 | |||
| 170337d4d5 | |||
| f0b27b3b37 | 
							
								
								
									
										6
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -1,8 +1,10 @@ | ||||
| * | ||||
| !src | ||||
| !src/** | ||||
| !docs | ||||
| !docs/** | ||||
| !doc | ||||
| !doc/** | ||||
| !test | ||||
| !test/** | ||||
| !Makefile | ||||
| !.gitignore | ||||
| !README.md | ||||
|   | ||||
							
								
								
									
										97
									
								
								Makefile
									
									
									
									
									
								
							
							
						
						
									
										97
									
								
								Makefile
									
									
									
									
									
								
							| @@ -1,26 +1,85 @@ | ||||
|  | ||||
| CC=gcc | ||||
| CFLAGS=-std=gnu11 -Wall -Wno-unused-but-set-variable | ||||
| LIBS=-lssl -lcrypto -lmagic -lz -lmaxminddb -lbrotlienc | ||||
| CFLAGS=-std=gnu11 -Wno-unused-but-set-variable -D_DEFAULT_SOURCE -D_BSD_SOURCE -D_SVID_SOURCE -D_POSIX_C_SOURCE=200809L | ||||
| LDFLAGS=-lssl -lcrypto -lmagic -lz -lmaxminddb -lbrotlienc | ||||
|  | ||||
| DEBIAN_OPTS=-D CACHE_MAGIC_FILE="\"/usr/share/file/magic.mgc\"" -D PHP_FPM_SOCKET="\"/var/run/php/php7.4-fpm.sock\"" | ||||
| DEBIAN_OPTS=-D CACHE_MAGIC_FILE="\"/usr/share/file/magic.mgc\"" -D PHP_FPM_SOCKET="\"/var/run/php/php8.2-fpm.sock\"" | ||||
|  | ||||
| .PHONY: all prod debug default debian permit clean test | ||||
| all: prod | ||||
| default: bin bin/lib bin/sesimos | ||||
|  | ||||
| prod: CFLAGS += -O3 | ||||
| prod: default | ||||
|  | ||||
| debug: CFLAGS += -Wall -pedantic | ||||
| debug: default | ||||
|  | ||||
| debian: CFLAGS += $(DEBIAN_OPTS) | ||||
| debian: prod | ||||
|  | ||||
| test: CFLAGS += -include test/mock_*.h | ||||
| test: bin bin/test | ||||
| 	bin/test | ||||
|  | ||||
|  | ||||
| bin: | ||||
| 	mkdir -p bin | ||||
|  | ||||
| bin/lib: | ||||
| 	mkdir -p bin/lib | ||||
|  | ||||
|  | ||||
| bin/test: test/mock_*.c test/test_*.c src/lib/utils.c src/lib/sock.c | ||||
| 	$(CC) -o $@ $(CFLAGS) $^ -lcriterion | ||||
|  | ||||
|  | ||||
| bin/%.o: src/%.c | ||||
| 	$(CC) -c -o $@ $(CFLAGS) $< | ||||
|  | ||||
| bin/lib/%.o: src/lib/%.c | ||||
| 	$(CC) -c -o $@ $(CFLAGS) $< | ||||
|  | ||||
| bin/sesimos: bin/server.o bin/client.o \ | ||||
| 			 bin/lib/cache.o bin/lib/compress.o bin/lib/config.o bin/lib/fastcgi.o bin/lib/geoip.o \ | ||||
| 			 bin/lib/http.o bin/lib/http_static.o bin/lib/rev_proxy.o bin/lib/sock.o bin/lib/uri.o \ | ||||
| 		     bin/lib/utils.o bin/lib/websocket.o | ||||
| 	$(CC) -o $@ $^ $(CFLAGS) $(LDFLAGS) | ||||
|  | ||||
|  | ||||
| bin/server.o: src/server.h src/defs.h src/client.h src/lib/cache.h src/lib/config.h src/lib/sock.h \ | ||||
|               src/lib/rev_proxy.h src/lib/geoip.h src/lib/utils.h | ||||
|  | ||||
| bin/client.o: src/client.h src/defs.h src/server.h src/lib/utils.h src/lib/config.h src/lib/sock.h \ | ||||
|               src/lib/http.h src/lib/rev_proxy.h src/lib/fastcgi.h src/lib/cache.h src/lib/geoip.h src/lib/compress.h \ | ||||
|               src/lib/websocket.h | ||||
|  | ||||
| bin/lib/cache.o: src/lib/cache.h src/lib/utils.h src/lib/uri.h src/lib/compress.h | ||||
|  | ||||
| bin/lib/compress.o: src/lib/compress.h | ||||
|  | ||||
| bin/lib/config.o: src/lib/config.h src/lib/utils.h src/lib/uri.h | ||||
|  | ||||
| bin/lib/fastcgi.o: src/lib/fastcgi.h src/server.h src/lib/utils.h src/lib/compress.h src/lib/http.h \ | ||||
|                    src/lib/uri.h src/lib/include/fastcgi.h | ||||
|  | ||||
| bin/lib/geoip.o: src/lib/geoip.h | ||||
|  | ||||
| bin/lib/http.o: src/lib/http.h src/lib/utils.h src/lib/compress.h src/lib/sock.h | ||||
|  | ||||
| bin/lib/rev_proxy.o: src/lib/rev_proxy.h src/defs.h src/server.h src/lib/compress.h | ||||
|  | ||||
| bin/lib/sock.o: src/lib/sock.h | ||||
|  | ||||
| bin/lib/uri.o: src/lib/uri.h src/lib/utils.h | ||||
|  | ||||
| bin/lib/utils.o: src/lib/utils.h | ||||
|  | ||||
| bin/lib/websocket.o: src/lib/websocket.h src/defs.h src/lib/utils.h src/lib/sock.h | ||||
|  | ||||
| packages: | ||||
| 	@echo "Installing packages..." | ||||
| 	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" | ||||
| 	sudo setcap 'cap_net_bind_service=+ep' "$(shell pwd)/bin/sesimos" | ||||
|  | ||||
| compile: | ||||
| 	@mkdir -p bin | ||||
| 	$(CC) src/lib/*.c -o bin/libnecrondaserver.so --shared -fPIC $(CFLAGS) $(LIBS) | ||||
| 	$(CC) src/server.c src/client.c -o bin/necronda-server $(CFLAGS) $(LIBS) \ | ||||
| 		-Lbin -lnecrondaserver -Wl,-rpath=$(shell pwd)/bin | ||||
|  | ||||
| compile-prod: | ||||
| 	@mkdir -p bin | ||||
| 	$(CC) src/lib/*.c -o bin/libnecrondaserver.so --shared -fPIC $(CFLAGS) $(LIBS) $(DEBIAN_OPTS) -O3 | ||||
| 	$(CC) src/server.c src/client.c -o bin/necronda-server $(CFLAGS) $(LIBS) $(DEBIAN_OPTS) -O3 \ | ||||
| 		-Lbin -lnecrondaserver -Wl,-rpath=$(shell pwd)/bin | ||||
| clean: | ||||
| 	rm -rf bin/* | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
|  | ||||
| Necronda web server | ||||
| =================== | ||||
| Sesimos – Secure, simple, modern web server | ||||
| =========================================== | ||||
|  | ||||
| ## Features | ||||
|  | ||||
| @@ -10,7 +10,7 @@ Necronda web server | ||||
|   * File compression ([gzip](https://www.gzip.org/), [Brotli](https://www.brotli.org/)) | ||||
|   * Disk cache for compressed files | ||||
| * Reverse proxy for other HTTP and HTTPS servers | ||||
|   * Transparent WebSocket reverse proxy **[WIP]** | ||||
|   * Transparent WebSocket reverse proxy | ||||
| * FastCGI support (e.g. [PHP-FPM](https://php-fpm.org/)) | ||||
|   * Automatic path info detection (e.g. `/my/file/extra/path` -> script: `/my/file.php`, path info: `extra/path`) | ||||
| * Support for [MaxMind's GeoIP Database](https://www.maxmind.com/en/geoip2-services-and-databases) | ||||
| @@ -21,7 +21,7 @@ Necronda web server | ||||
|  | ||||
| ## Configuration | ||||
|  | ||||
| See [docs/example.conf](docs/example.conf) for more details. | ||||
| See [doc/example.conf](doc/example.conf) for more details. | ||||
|  | ||||
|  | ||||
| ### Global directives | ||||
|   | ||||
							
								
								
									
										138
									
								
								src/client.c
									
									
									
									
									
								
							
							
						
						
									
										138
									
								
								src/client.c
									
									
									
									
									
								
							| @@ -1,12 +1,13 @@ | ||||
| /** | ||||
|  * Necronda Web Server | ||||
|  * Client connection and request handlers | ||||
|  * src/client.c | ||||
|  * Lorenz Stechauner, 2020-12-03 | ||||
|  * sesimos - secure, simple, modern web server | ||||
|  * @brief Client connection and request handlers | ||||
|  * @file src/client.c | ||||
|  * @author Lorenz Stechauner | ||||
|  * @date 2020-12-03 | ||||
|  */ | ||||
|  | ||||
| #include "defs.h" | ||||
| #include "client.h" | ||||
| #include "necronda.h" | ||||
| #include "server.h" | ||||
|  | ||||
| #include "lib/utils.h" | ||||
| @@ -18,9 +19,9 @@ | ||||
| #include "lib/cache.h" | ||||
| #include "lib/geoip.h" | ||||
| #include "lib/compress.h" | ||||
| #include "lib/websocket.h" | ||||
|  | ||||
| #include <string.h> | ||||
| #include <sys/select.h> | ||||
| #include <errno.h> | ||||
| #include <unistd.h> | ||||
| #include <openssl/ssl.h> | ||||
| @@ -28,13 +29,12 @@ | ||||
| #include <signal.h> | ||||
| #include <arpa/inet.h> | ||||
|  | ||||
| int server_keep_alive = 1; | ||||
|  | ||||
| volatile sig_atomic_t server_keep_alive = 1; | ||||
| 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 < CONFIG_MAX_HOST_CONFIG; i++) { | ||||
| @@ -49,15 +49,10 @@ host_config *get_host_config(const char *host) { | ||||
|     return NULL; | ||||
| } | ||||
|  | ||||
| void client_terminate() { | ||||
| void client_terminate(int _) { | ||||
|     server_keep_alive = 0; | ||||
| } | ||||
|  | ||||
| int client_websocket_handler() { | ||||
|     // TODO implement client_websocket_handler | ||||
|     return 0; | ||||
| } | ||||
|  | ||||
| int client_request_handler(sock *client, unsigned long client_num, unsigned int req_num) { | ||||
|     struct timespec begin, end; | ||||
|     long ret; | ||||
| @@ -87,16 +82,12 @@ int client_request_handler(sock *client, unsigned long client_num, unsigned int | ||||
|     http_status custom_status; | ||||
|  | ||||
|     http_res res = {.version = "1.1", .status = http_get_status(501), .hdr.field_num = 0, .hdr.last_field_num = -1}; | ||||
|     http_status_ctx ctx = {.status = 0, .origin = NONE}; | ||||
|     http_status_ctx ctx = {.status = 0, .origin = NONE, .ws_key = NULL}; | ||||
|  | ||||
|     clock_gettime(CLOCK_MONOTONIC, &begin); | ||||
|  | ||||
|     fd_set socket_fds; | ||||
|     FD_ZERO(&socket_fds); | ||||
|     FD_SET(client->socket, &socket_fds); | ||||
|     client_timeout.tv_sec = CLIENT_TIMEOUT; | ||||
|     client_timeout.tv_usec = 0; | ||||
|     ret = select(client->socket + 1, &socket_fds, NULL, NULL, &client_timeout); | ||||
|     ret = sock_poll_read(&client, NULL, NULL, 1, NULL, NULL, CLIENT_TIMEOUT * 1000); | ||||
|  | ||||
|     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) { | ||||
| @@ -130,7 +121,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)); | ||||
|     client_keep_alive = (hdr_connection != NULL && (strstr(hdr_connection, "keep-alive") != NULL || strstr(hdr_connection, "Keep-Alive") != NULL)); | ||||
|     host_ptr = http_get_header_field(&req.hdr, "Host"); | ||||
|     if (host_ptr != NULL && strlen(host_ptr) > 255) { | ||||
|         host[0] = 0; | ||||
| @@ -188,8 +179,7 @@ int client_request_handler(sock *client, unsigned long client_num, unsigned int | ||||
|         int change_proto = strncmp(uri.uri, "/.well-known/", 13) != 0 && !client->enc; | ||||
|         if (strcmp(uri.uri, buf0) != 0 || change_proto) { | ||||
|             res.status = http_get_status(308); | ||||
|             size = sizeof(buf0); | ||||
|             url_encode(uri.uri, buf0, &size); | ||||
|             size = url_encode(uri.uri, strlen(uri.uri), buf0, sizeof(buf0)); | ||||
|             if (change_proto) { | ||||
|                 p_len = snprintf(buf1, sizeof(buf1), "https://%s%s", host, buf0); | ||||
|                 if (p_len < 0 || p_len >= sizeof(buf1)) { | ||||
| @@ -210,12 +200,6 @@ int client_request_handler(sock *client, unsigned long client_num, unsigned int | ||||
|         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 (strcmp(req.method, "TRACE") == 0) { | ||||
|             res.status = http_get_status(200); | ||||
| @@ -263,7 +247,7 @@ int client_request_handler(sock *client, unsigned long client_num, unsigned int | ||||
|                 goto respond; | ||||
|             } | ||||
|  | ||||
|             if (http_get_header_field(&req.hdr, "Content-Length") != NULL) { | ||||
|             if (http_get_header_field(&req.hdr, "Content-Length") != NULL || http_get_header_field(&req.hdr, "Transfer-Encoding") != NULL) { | ||||
|                 res.status = http_get_status(400); | ||||
|                 sprintf(err_msg, "A GET request must not contain a payload"); | ||||
|                 goto respond; | ||||
| @@ -384,7 +368,7 @@ int client_request_handler(sock *client, unsigned long client_num, unsigned int | ||||
|         } else { | ||||
|             int mode; | ||||
|             if (strcmp(uri.filename + strlen(uri.filename) - 4, ".ncr") == 0) { | ||||
|                 mode = FASTCGI_NECRONDA; | ||||
|                 mode = FASTCGI_SESIMOS; | ||||
|             } else if (strcmp(uri.filename + strlen(uri.filename) - 4, ".php") == 0) { | ||||
|                 mode = FASTCGI_PHP; | ||||
|             } else { | ||||
| @@ -406,18 +390,23 @@ int client_request_handler(sock *client, unsigned long client_num, unsigned int | ||||
|             } | ||||
|  | ||||
|             const char *client_content_length = http_get_header_field(&req.hdr, "Content-Length"); | ||||
|             const char *client_transfer_encoding = http_get_header_field(&req.hdr, "Transfer-Encoding"); | ||||
|             if (client_content_length != NULL) { | ||||
|                 unsigned long client_content_len = strtoul(client_content_length, NULL, 10); | ||||
|                 ret = fastcgi_receive(&fcgi_conn, client, client_content_len); | ||||
|                 if (ret != 0) { | ||||
|                     if (ret < 0) { | ||||
|                         goto abort; | ||||
|                     } else { | ||||
|                         sprintf(err_msg, "Unable to communicate with FastCGI socket."); | ||||
|                     } | ||||
|                     res.status = http_get_status(502); | ||||
|                     goto respond; | ||||
|             } else if (client_transfer_encoding != NULL && strstr(client_transfer_encoding, "chunked") != NULL) { | ||||
|                 ret = fastcgi_receive_chunked(&fcgi_conn, client); | ||||
|             } else { | ||||
|                 ret = 0; | ||||
|             } | ||||
|             if (ret != 0) { | ||||
|                 if (ret < 0) { | ||||
|                     goto abort; | ||||
|                 } else { | ||||
|                     sprintf(err_msg, "Unable to communicate with FastCGI socket."); | ||||
|                 } | ||||
|                 res.status = http_get_status(502); | ||||
|                 goto respond; | ||||
|             } | ||||
|             fastcgi_close_stdin(&fcgi_conn); | ||||
|  | ||||
| @@ -493,6 +482,24 @@ int client_request_handler(sock *client, unsigned long client_num, unsigned int | ||||
|         ret = rev_proxy_init(&req, &res, &ctx, conf, client, &custom_status, err_msg); | ||||
|         use_rev_proxy = (ret == 0); | ||||
|  | ||||
|         if (res.status->code == 101) { | ||||
|             const char *connection = http_get_header_field(&res.hdr, "Connection"); | ||||
|             const char *upgrade = http_get_header_field(&res.hdr, "Upgrade"); | ||||
|             if (connection != NULL && upgrade != NULL && | ||||
|                 (strstr(connection, "upgrade") != NULL || strstr(connection, "Upgrade") != NULL) && | ||||
|                 strcmp(upgrade, "websocket") == 0) | ||||
|             { | ||||
|                 const char *ws_accept = http_get_header_field(&res.hdr, "Sec-WebSocket-Accept"); | ||||
|                 if (ws_calc_accept_key(ctx.ws_key, buf0) == 0) { | ||||
|                     use_rev_proxy = (strcmp(buf0, ws_accept) == 0) ? 2 : 1; | ||||
|                 } | ||||
|             } else { | ||||
|                 ctx.status = 101; | ||||
|                 ctx.origin = INTERNAL; | ||||
|                 res.status = http_get_status(501); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         // Let 300 be formatted by origin server | ||||
|         if (use_rev_proxy && res.status->code >= 301 && res.status->code < 600) { | ||||
|             const char *content_type = http_get_header_field(&res.hdr, "Content-Type"); | ||||
| @@ -501,8 +508,10 @@ int client_request_handler(sock *client, unsigned long client_num, unsigned int | ||||
|             if (content_encoding == NULL && content_type != NULL && content_length_f != NULL && strncmp(content_type, "text/html", 9) == 0) { | ||||
|                 long content_len = strtol(content_length_f, NULL, 10); | ||||
|                 if (content_len <= sizeof(msg_content) - 1) { | ||||
|                     ctx.status = res.status->code; | ||||
|                     ctx.origin = res.status->code >= 400 ? SERVER : NONE; | ||||
|                     if (ctx.status != 101) { | ||||
|                         ctx.status = res.status->code; | ||||
|                         ctx.origin = res.status->code >= 400 ? SERVER : NONE; | ||||
|                     } | ||||
|                     use_rev_proxy = 0; | ||||
|                     rev_proxy_dump(msg_content, content_len); | ||||
|                 } | ||||
| @@ -609,16 +618,19 @@ int client_request_handler(sock *client, unsigned long client_num, unsigned int | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     const char *conn = http_get_header_field(&res.hdr, "Connection"); | ||||
|     int close_proxy = (conn == NULL || (strcmp(conn, "keep-alive") != 0 && strcmp(conn, "Keep-Alive") != 0)); | ||||
|     http_remove_header_field(&res.hdr, "Connection", HTTP_REMOVE_ALL); | ||||
|     http_remove_header_field(&res.hdr, "Keep-Alive", HTTP_REMOVE_ALL); | ||||
|     if (server_keep_alive && client_keep_alive) { | ||||
|         http_add_header_field(&res.hdr, "Connection", "keep-alive"); | ||||
|         sprintf(buf0, "timeout=%i, max=%i", CLIENT_TIMEOUT, REQ_PER_CONNECTION); | ||||
|         http_add_header_field(&res.hdr, "Keep-Alive", buf0); | ||||
|     } else { | ||||
|         http_add_header_field(&res.hdr, "Connection", "close"); | ||||
|     int close_proxy = 0; | ||||
|     if (use_rev_proxy != 2) { | ||||
|         const char *conn = http_get_header_field(&res.hdr, "Connection"); | ||||
|         close_proxy = (conn == NULL || (strstr(conn, "keep-alive") == NULL && strstr(conn, "Keep-Alive") == NULL)); | ||||
|         http_remove_header_field(&res.hdr, "Connection", HTTP_REMOVE_ALL); | ||||
|         http_remove_header_field(&res.hdr, "Keep-Alive", HTTP_REMOVE_ALL); | ||||
|         if (server_keep_alive && client_keep_alive) { | ||||
|             http_add_header_field(&res.hdr, "Connection", "keep-alive"); | ||||
|             sprintf(buf0, "timeout=%i, max=%i", CLIENT_TIMEOUT, REQ_PER_CONNECTION); | ||||
|             http_add_header_field(&res.hdr, "Keep-Alive", buf0); | ||||
|         } else { | ||||
|             http_add_header_field(&res.hdr, "Connection", "close"); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     http_send_response(client, &res); | ||||
| @@ -631,7 +643,17 @@ int client_request_handler(sock *client, unsigned long client_num, unsigned int | ||||
|  | ||||
|     // TODO access/error log file | ||||
|  | ||||
|     if (strcmp(req.method, "HEAD") != 0) { | ||||
|     if (use_rev_proxy == 2) { | ||||
|         // WebSocket | ||||
|         print("Upgrading connection to WebSocket connection"); | ||||
|         ret = ws_handle_connection(client, &rev_proxy); | ||||
|         if (ret != 0) { | ||||
|             client_keep_alive = 0; | ||||
|             close_proxy = 1; | ||||
|         } | ||||
|         print("WebSocket connection closed"); | ||||
|     } else if (strcmp(req.method, "HEAD") != 0) { | ||||
|         // default response | ||||
|         unsigned long snd_len = 0; | ||||
|         unsigned long len; | ||||
|         if (msg_buf[0] != 0) { | ||||
| @@ -655,7 +677,7 @@ int client_request_handler(sock *client, unsigned long client_num, unsigned int | ||||
|             } | ||||
|         } else if (use_fastcgi) { | ||||
|             const char *transfer_encoding = http_get_header_field(&res.hdr, "Transfer-Encoding"); | ||||
|             int chunked = (transfer_encoding != NULL && strcmp(transfer_encoding, "chunked") == 0); | ||||
|             int chunked = (transfer_encoding != NULL && strstr(transfer_encoding, "chunked") != NULL); | ||||
|  | ||||
|             int flags = (chunked ? FASTCGI_CHUNKED : 0) | (use_fastcgi & (FASTCGI_COMPRESS | FASTCGI_COMPRESS_HOLD)); | ||||
|             ret = fastcgi_send(&fcgi_conn, client, flags); | ||||
| @@ -696,12 +718,6 @@ int client_request_handler(sock *client, unsigned long client_num, unsigned int | ||||
|     } | ||||
|     http_free_req(&req); | ||||
|     http_free_res(&res); | ||||
|     if (client->buf != NULL) { | ||||
|         free(client->buf); | ||||
|         client->buf = NULL; | ||||
|         client->buf_off = 0; | ||||
|         client->buf_len = 0; | ||||
|     } | ||||
|     return !client_keep_alive; | ||||
| } | ||||
|  | ||||
|   | ||||
							
								
								
									
										15
									
								
								src/client.h
									
									
									
									
									
								
							
							
						
						
									
										15
									
								
								src/client.h
									
									
									
									
									
								
							| @@ -1,12 +1,13 @@ | ||||
| /** | ||||
|  * Necronda Web Server | ||||
|  * Client connection and request handlers (header file) | ||||
|  * src/client.h | ||||
|  * Lorenz Stechauner, 2022-08-16 | ||||
|  * sesimos - secure, simple, modern web server | ||||
|  * @brief Client connection and request handlers (header file) | ||||
|  * @file src/client.h | ||||
|  * @author Lorenz Stechauner | ||||
|  * @date 2022-08-16 | ||||
|  */ | ||||
|  | ||||
| #ifndef NECRONDA_SERVER_NECRONDA_CLIENT_H | ||||
| #define NECRONDA_SERVER_NECRONDA_CLIENT_H | ||||
| #ifndef SESIMOS_CLIENT_H | ||||
| #define SESIMOS_CLIENT_H | ||||
|  | ||||
| #include "lib/config.h" | ||||
| #include "lib/sock.h" | ||||
| @@ -17,4 +18,4 @@ host_config *get_host_config(const char *host); | ||||
|  | ||||
| int client_handler(sock *client, unsigned long client_num, struct sockaddr_in6 *client_addr); | ||||
|  | ||||
| #endif //NECRONDA_SERVER_NECRONDA_CLIENT_H | ||||
| #endif //SESIMOS_CLIENT_H | ||||
|   | ||||
							
								
								
									
										26
									
								
								src/defs.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								src/defs.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,26 @@ | ||||
| /** | ||||
|  * sesimos - secure, simple, modern web server | ||||
|  * @brief Definitions | ||||
|  * @file src/defs.h | ||||
|  * @author Lorenz Stechauner | ||||
|  * @date 2021-05-04 | ||||
|  */ | ||||
|  | ||||
| #ifndef SESIMOS_DEF_H | ||||
| #define SESIMOS_DEF_H | ||||
|  | ||||
| #define SERVER_VERSION "4.6" | ||||
| #define SERVER_STR "Sesimos/" SERVER_VERSION | ||||
| #define SERVER_STR_HTML "Sesimos web server " SERVER_VERSION | ||||
|  | ||||
| #define CHUNK_SIZE 8192 | ||||
|  | ||||
| #ifndef DEFAULT_HOST | ||||
| #   define DEFAULT_HOST "www.necronda.net" | ||||
| #endif | ||||
|  | ||||
| #ifndef SERVER_NAME | ||||
| #   define SERVER_NAME DEFAULT_HOST | ||||
| #endif | ||||
|  | ||||
| #endif //SESIMOS_DEF_H | ||||
| @@ -1,8 +1,9 @@ | ||||
| /** | ||||
|  * Necronda Web Server | ||||
|  * File cache implementation | ||||
|  * src/lib/cache.c | ||||
|  * Lorenz Stechauner, 2020-12-19 | ||||
|  * sesimos - secure, simple, modern web server | ||||
|  * @brief File cache implementation | ||||
|  * @file src/lib/cache.c | ||||
|  * @author Lorenz Stechauner | ||||
|  * @date 2020-12-19 | ||||
|  */ | ||||
|  | ||||
| #include "cache.h" | ||||
| @@ -16,14 +17,14 @@ | ||||
| #include <string.h> | ||||
| #include <errno.h> | ||||
| #include <signal.h> | ||||
| #include <openssl/sha.h> | ||||
| #include <malloc.h> | ||||
| #include <openssl/evp.h> | ||||
|  | ||||
|  | ||||
| int cache_continue = 1; | ||||
| magic_t magic; | ||||
| cache_entry *cache; | ||||
|  | ||||
| int magic_init() { | ||||
| int magic_init(void) { | ||||
|     magic = magic_open(MAGIC_MIME); | ||||
|     if (magic == NULL) { | ||||
|         fprintf(stderr, ERR_STR "Unable to open magic cookie: %s" CLR_STR "\n", strerror(errno)); | ||||
| @@ -36,11 +37,11 @@ int magic_init() { | ||||
|     return 0; | ||||
| } | ||||
|  | ||||
| void cache_process_term() { | ||||
| void cache_process_term(int _) { | ||||
|     cache_continue = 0; | ||||
| } | ||||
|  | ||||
| int cache_process() { | ||||
| int cache_process(void) { | ||||
|     signal(SIGINT, cache_process_term); | ||||
|     signal(SIGTERM, cache_process_term); | ||||
|  | ||||
| @@ -58,17 +59,17 @@ int cache_process() { | ||||
|     } | ||||
|     cache = shm_rw; | ||||
|  | ||||
|     if (mkdir("/var/necronda/", 0755) < 0 && errno != EEXIST) { | ||||
|         fprintf(stderr, ERR_STR "Unable to create directory '/var/necronda/': %s" CLR_STR "\n", strerror(errno)); | ||||
|     if (mkdir("/var/sesimos/", 0755) < 0 && errno != EEXIST) { | ||||
|         fprintf(stderr, ERR_STR "Unable to create directory '/var/sesimos/': %s" CLR_STR "\n", strerror(errno)); | ||||
|         return -3; | ||||
|     } | ||||
|  | ||||
|     if (mkdir("/var/necronda/server/", 0755) < 0 && errno != EEXIST) { | ||||
|         fprintf(stderr, ERR_STR "Unable to create directory '/var/necronda/server/': %s" CLR_STR "\n", strerror(errno)); | ||||
|     if (mkdir("/var/sesimos/server/", 0755) < 0 && errno != EEXIST) { | ||||
|         fprintf(stderr, ERR_STR "Unable to create directory '/var/sesimos/server/': %s" CLR_STR "\n", strerror(errno)); | ||||
|         return -3; | ||||
|     } | ||||
|  | ||||
|     FILE *cache_file = fopen("/var/necronda/server/cache", "rb"); | ||||
|     FILE *cache_file = fopen("/var/sesimos/server/cache", "rb"); | ||||
|     if (cache_file != NULL) { | ||||
|         fread(cache, sizeof(cache_entry), CACHE_ENTRIES, cache_file); | ||||
|         fclose(cache_file); | ||||
| @@ -79,14 +80,12 @@ int cache_process() { | ||||
|     } | ||||
|  | ||||
|     FILE *file; | ||||
|     char *buf = malloc(CACHE_BUF_SIZE); | ||||
|     char *comp_buf = malloc(CACHE_BUF_SIZE); | ||||
|     char filename_comp_gz[256]; | ||||
|     char filename_comp_br[256]; | ||||
|     char buf[CACHE_BUF_SIZE], comp_buf[CACHE_BUF_SIZE], filename_comp_gz[256], filename_comp_br[256]; | ||||
|     unsigned long read; | ||||
|     int compress; | ||||
|     SHA_CTX ctx; | ||||
|     unsigned char hash[SHA_DIGEST_LENGTH]; | ||||
|     EVP_MD_CTX *ctx; | ||||
|     unsigned char hash[EVP_MAX_MD_SIZE]; | ||||
|     unsigned int md_len; | ||||
|     int cache_changed = 0; | ||||
|     int p_len_gz, p_len_br; | ||||
|     int ret; | ||||
| @@ -95,7 +94,9 @@ int cache_process() { | ||||
|             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); | ||||
|  | ||||
|                 ctx = EVP_MD_CTX_new(); | ||||
|                 EVP_DigestInit(ctx, EVP_sha1()); | ||||
|                 file = fopen(cache[i].filename, "rb"); | ||||
|                 compress = mime_is_compressible(cache[i].meta.type); | ||||
|  | ||||
| @@ -103,13 +104,13 @@ int cache_process() { | ||||
|                 FILE *comp_file_gz = NULL; | ||||
|                 FILE *comp_file_br = NULL; | ||||
|                 if (compress) { | ||||
|                     sprintf(buf, "%.*s/.necronda-server", cache[i].webroot_len, cache[i].filename); | ||||
|                     sprintf(buf, "%.*s/.sesimos", cache[i].webroot_len, cache[i].filename); | ||||
|                     if (mkdir(buf, 0755) != 0 && errno != EEXIST) { | ||||
|                         fprintf(stderr, ERR_STR "Unable to create directory %s: %s" CLR_STR "\n", buf, strerror(errno)); | ||||
|                         goto comp_err; | ||||
|                     } | ||||
|  | ||||
|                     sprintf(buf, "%.*s/.necronda-server/cache", cache[i].webroot_len, cache[i].filename); | ||||
|                     sprintf(buf, "%.*s/.sesimos/cache", cache[i].webroot_len, cache[i].filename); | ||||
|                     if (mkdir(buf, 0700) != 0 && errno != EEXIST) { | ||||
|                         fprintf(stderr, ERR_STR "Unable to create directory %s: %s" CLR_STR "\n", buf, strerror(errno)); | ||||
|                         goto comp_err; | ||||
| @@ -124,16 +125,13 @@ int cache_process() { | ||||
|                     buf[strlen(rel_path)] = 0; | ||||
|  | ||||
|                     p_len_gz = snprintf(filename_comp_gz, sizeof(filename_comp_gz), | ||||
|                                         "%.*s/.necronda-server/cache/%s.gz", | ||||
|                                         "%.*s/.sesimos/cache/%s.gz", | ||||
|                                         cache[i].webroot_len, cache[i].filename, buf); | ||||
|                     p_len_br = snprintf(filename_comp_br, sizeof(filename_comp_br), | ||||
|                                         "%.*s/.necronda-server/cache/%s.br", | ||||
|                                         "%.*s/.sesimos/cache/%s.br", | ||||
|                                         cache[i].webroot_len, cache[i].filename, buf); | ||||
|                     if (p_len_gz < 0 || p_len_gz >= sizeof(filename_comp_gz) || | ||||
|                         p_len_br < 0 || p_len_br >= sizeof(filename_comp_br)) | ||||
|                     { | ||||
|                         fprintf(stderr, ERR_STR "Unable to open cached file: " | ||||
|                                                 "File name for compressed file too long" CLR_STR "\n"); | ||||
|                     if (p_len_gz < 0 || p_len_gz >= sizeof(filename_comp_gz) || p_len_br < 0 || p_len_br >= sizeof(filename_comp_br)) { | ||||
|                         fprintf(stderr, ERR_STR "Unable to open cached file: File name for compressed file too long" CLR_STR "\n"); | ||||
|                         goto comp_err; | ||||
|                     } | ||||
|  | ||||
| @@ -157,21 +155,19 @@ int cache_process() { | ||||
|                 } | ||||
|  | ||||
|                 while ((read = fread(buf, 1, CACHE_BUF_SIZE, file)) > 0) { | ||||
|                     SHA1_Update(&ctx, buf, read); | ||||
|                     EVP_DigestUpdate(ctx, buf, read); | ||||
|                     if (compress) { | ||||
|                         unsigned long avail_in, avail_out; | ||||
|                         avail_in = read; | ||||
|                         do { | ||||
|                             avail_out = CACHE_BUF_SIZE; | ||||
|                             compress_compress_mode(&comp_ctx, COMPRESS_GZ,buf + read - avail_in, &avail_in, | ||||
|                                                    comp_buf, &avail_out, feof(file)); | ||||
|                             compress_compress_mode(&comp_ctx, COMPRESS_GZ,buf + read - avail_in, &avail_in, comp_buf, &avail_out, feof(file)); | ||||
|                             fwrite(comp_buf, 1, CACHE_BUF_SIZE - avail_out, comp_file_gz); | ||||
|                         } while (avail_in != 0 || avail_out != CACHE_BUF_SIZE); | ||||
|                         avail_in = read; | ||||
|                         do { | ||||
|                             avail_out = CACHE_BUF_SIZE; | ||||
|                             compress_compress_mode(&comp_ctx, COMPRESS_BR, buf + read - avail_in, &avail_in, | ||||
|                                                    comp_buf, &avail_out, feof(file)); | ||||
|                             compress_compress_mode(&comp_ctx, COMPRESS_BR, buf + read - avail_in, &avail_in, comp_buf, &avail_out, feof(file)); | ||||
|                             fwrite(comp_buf, 1, CACHE_BUF_SIZE - avail_out, comp_file_br); | ||||
|                         } while (avail_in != 0 || avail_out != CACHE_BUF_SIZE); | ||||
|                     } | ||||
| @@ -188,9 +184,12 @@ int cache_process() { | ||||
|                     memset(cache[i].meta.filename_comp_gz, 0, sizeof(cache[i].meta.filename_comp_gz)); | ||||
|                     memset(cache[i].meta.filename_comp_br, 0, sizeof(cache[i].meta.filename_comp_br)); | ||||
|                 } | ||||
|                 SHA1_Final(hash, &ctx); | ||||
|  | ||||
|                 EVP_DigestFinal(ctx, hash, &md_len); | ||||
|                 EVP_MD_CTX_free(ctx); | ||||
|  | ||||
|                 memset(cache[i].meta.etag, 0, sizeof(cache[i].meta.etag)); | ||||
|                 for (int j = 0; j < SHA_DIGEST_LENGTH; j++) { | ||||
|                 for (int j = 0; j < md_len; j++) { | ||||
|                     sprintf(cache[i].meta.etag + j * 2, "%02x", hash[j]); | ||||
|                 } | ||||
|                 fclose(file); | ||||
| @@ -202,11 +201,9 @@ int cache_process() { | ||||
|  | ||||
|         if (cache_changed) { | ||||
|             cache_changed = 0; | ||||
|             cache_file = fopen("/var/necronda/server/cache", "wb"); | ||||
|             cache_file = fopen("/var/sesimos/server/cache", "wb"); | ||||
|             if (cache_file == NULL) { | ||||
|                 fprintf(stderr, ERR_STR "Unable to open cache file: %s" CLR_STR "\n", strerror(errno)); | ||||
|                 free(buf); | ||||
|                 free(comp_buf); | ||||
|                 return -1; | ||||
|             } | ||||
|             fwrite(cache, sizeof(cache_entry), CACHE_ENTRIES, cache_file); | ||||
| @@ -215,12 +212,11 @@ int cache_process() { | ||||
|             sleep(1); | ||||
|         } | ||||
|     } | ||||
|     free(buf); | ||||
|     free(comp_buf); | ||||
|  | ||||
|     return 0; | ||||
| } | ||||
|  | ||||
| int cache_init() { | ||||
| int cache_init(void) { | ||||
|     if (magic_init() != 0) { | ||||
|         return -1; | ||||
|     } | ||||
| @@ -266,7 +262,7 @@ int cache_init() { | ||||
|     } | ||||
| } | ||||
|  | ||||
| int cache_unload() { | ||||
| int cache_unload(void) { | ||||
|     int shm_id = shmget(CACHE_SHM_KEY, 0, 0); | ||||
|     if (shm_id < 0) { | ||||
|         fprintf(stderr, ERR_STR "Unable to get cache shared memory id: %s" CLR_STR "\n", strerror(errno)); | ||||
|   | ||||
| @@ -1,12 +1,13 @@ | ||||
| /** | ||||
|  * Necronda Web Server | ||||
|  * File cache implementation (header file) | ||||
|  * src/lib/cache.h | ||||
|  * Lorenz Stechauner, 2020-12-19 | ||||
|  * sesimos - secure, simple, modern web server | ||||
|  * @brief File cache implementation (header file) | ||||
|  * @file src/lib/cache.h | ||||
|  * @author Lorenz Stechauner | ||||
|  * @date 2020-12-19 | ||||
|  */ | ||||
|  | ||||
| #ifndef NECRONDA_SERVER_CACHE_H | ||||
| #define NECRONDA_SERVER_CACHE_H | ||||
| #ifndef SESIMOS_CACHE_H | ||||
| #define SESIMOS_CACHE_H | ||||
|  | ||||
| #include "uri.h" | ||||
|  | ||||
| @@ -30,15 +31,15 @@ extern cache_entry *cache; | ||||
|  | ||||
| extern int cache_continue; | ||||
|  | ||||
| int magic_init(); | ||||
| int magic_init(void); | ||||
|  | ||||
| void cache_process_term(); | ||||
| void cache_process_term(int _); | ||||
|  | ||||
| int cache_process(); | ||||
| int cache_process(void); | ||||
|  | ||||
| int cache_init(); | ||||
| int cache_init(void); | ||||
|  | ||||
| int cache_unload(); | ||||
| int cache_unload(void); | ||||
|  | ||||
| int cache_update_entry(int entry_num, const char *filename, const char *webroot); | ||||
|  | ||||
| @@ -46,4 +47,4 @@ int cache_filename_comp_invalid(const char *filename); | ||||
|  | ||||
| int uri_cache_init(http_uri *uri); | ||||
|  | ||||
| #endif //NECRONDA_SERVER_CACHE_H | ||||
| #endif //SESIMOS_CACHE_H | ||||
|   | ||||
| @@ -1,8 +1,9 @@ | ||||
| /** | ||||
|  * Necronda Web Server | ||||
|  * Compression interface | ||||
|  * src/lib/compress.c | ||||
|  * Lorenz Stechauner, 2021-05-05 | ||||
|  * sesimos - secure, simple, modern web server | ||||
|  * @brief Compression interface | ||||
|  * @file src/lib/compress.c | ||||
|  * @author Lorenz Stechauner | ||||
|  * @date 2021-05-05 | ||||
|  */ | ||||
|  | ||||
| #include "compress.h" | ||||
| @@ -10,6 +11,7 @@ | ||||
| #include <malloc.h> | ||||
| #include <errno.h> | ||||
|  | ||||
|  | ||||
| int compress_init(compress_ctx *ctx, int mode) { | ||||
|     ctx->gzip = NULL; | ||||
|     ctx->brotli = NULL; | ||||
|   | ||||
| @@ -1,12 +1,13 @@ | ||||
| /** | ||||
|  * Necronda Web Server | ||||
|  * Compression interface (header file) | ||||
|  * src/lib/compress.h | ||||
|  * Lorenz Stechauner, 2021-05-05 | ||||
|  * sesimos - secure, simple, modern web server | ||||
|  * @brief Compression interface (header file) | ||||
|  * @file src/lib/compress.h | ||||
|  * @author Lorenz Stechauner | ||||
|  * @date 2021-05-05 | ||||
|  */ | ||||
|  | ||||
| #ifndef NECRONDA_SERVER_COMPRESS_H | ||||
| #define NECRONDA_SERVER_COMPRESS_H | ||||
| #ifndef SESIMOS_COMPRESS_H | ||||
| #define SESIMOS_COMPRESS_H | ||||
|  | ||||
| #include <zlib.h> | ||||
| #include <brotli/encode.h> | ||||
| @@ -34,4 +35,4 @@ int compress_compress_mode(compress_ctx *ctx, int mode, const char *in, unsigned | ||||
|  | ||||
| int compress_free(compress_ctx *ctx); | ||||
|  | ||||
| #endif //NECRONDA_SERVER_COMPRESS_H | ||||
| #endif //SESIMOS_COMPRESS_H | ||||
|   | ||||
| @@ -1,8 +1,9 @@ | ||||
| /** | ||||
|  * Necronda Web Server | ||||
|  * Configuration file loader | ||||
|  * src/lib/config.c | ||||
|  * Lorenz Stechauner, 2021-01-05 | ||||
|  * sesimos - secure, simple, modern web server | ||||
|  * @brief Configuration file loader | ||||
|  * @file src/lib/config.c | ||||
|  * @author Lorenz Stechauner | ||||
|  * @date 2021-01-05 | ||||
|  */ | ||||
|  | ||||
| #include "config.h" | ||||
| @@ -15,10 +16,11 @@ | ||||
| #include <errno.h> | ||||
| #include <stdlib.h> | ||||
|  | ||||
|  | ||||
| t_config *config; | ||||
| char geoip_dir[256], dns_server[256]; | ||||
|  | ||||
| int config_init() { | ||||
| int config_init(void) { | ||||
|     int shm_id = shmget(CONFIG_SHM_KEY, sizeof(t_config), IPC_CREAT | IPC_EXCL | 0640); | ||||
|     if (shm_id < 0) { | ||||
|         fprintf(stderr, ERR_STR "Unable to create config shared memory: %s" CLR_STR "\n", strerror(errno)); | ||||
| @@ -44,7 +46,7 @@ int config_init() { | ||||
|     return 0; | ||||
| } | ||||
|  | ||||
| int config_unload() { | ||||
| int config_unload(void) { | ||||
|     int shm_id = shmget(CONFIG_SHM_KEY, 0, 0); | ||||
|     if (shm_id < 0) { | ||||
|         fprintf(stderr, ERR_STR "Unable to get config shared memory id: %s" CLR_STR "\n", strerror(errno)); | ||||
|   | ||||
| @@ -1,12 +1,13 @@ | ||||
| /** | ||||
|  * Necronda Web Server | ||||
|  * Configuration file loader (header file) | ||||
|  * src/lib/config.h | ||||
|  * Lorenz Stechauner, 2021-01-05 | ||||
|  * sesimos - secure, simple, modern web server | ||||
|  * @brief Configuration file loader (header file) | ||||
|  * @file src/lib/config.h | ||||
|  * @author Lorenz Stechauner | ||||
|  * @date 2021-01-05 | ||||
|  */ | ||||
|  | ||||
| #ifndef NECRONDA_SERVER_CONFIG_H | ||||
| #define NECRONDA_SERVER_CONFIG_H | ||||
| #ifndef SESIMOS_CONFIG_H | ||||
| #define SESIMOS_CONFIG_H | ||||
|  | ||||
| #include "uri.h" | ||||
|  | ||||
| @@ -19,7 +20,7 @@ | ||||
| #define CONFIG_TYPE_REVERSE_PROXY 2 | ||||
|  | ||||
| #ifndef DEFAULT_CONFIG_FILE | ||||
| #   define DEFAULT_CONFIG_FILE "/etc/necronda/server.conf" | ||||
| #   define DEFAULT_CONFIG_FILE "/etc/sesimos/server.conf" | ||||
| #endif | ||||
|  | ||||
|  | ||||
| @@ -55,10 +56,10 @@ typedef struct { | ||||
| extern t_config *config; | ||||
| extern char geoip_dir[256], dns_server[256]; | ||||
|  | ||||
| int config_init(); | ||||
| int config_init(void); | ||||
|  | ||||
| int config_load(const char *filename); | ||||
|  | ||||
| int config_unload(); | ||||
| int config_unload(void); | ||||
|  | ||||
| #endif //NECRONDA_SERVER_CONFIG_H | ||||
| #endif //SESIMOS_CONFIG_H | ||||
|   | ||||
| @@ -1,8 +1,9 @@ | ||||
| /** | ||||
|  * Necronda Web Server | ||||
|  * FastCGI interface implementation | ||||
|  * src/lib/fastcgi.c | ||||
|  * Lorenz Stechauner, 2020-12-26 | ||||
|  * sesimos - secure, simple, modern web server | ||||
|  * @brief FastCGI interface implementation | ||||
|  * @file src/lib/fastcgi.c | ||||
|  * @author Lorenz Stechauner | ||||
|  * @date 2020-12-26 | ||||
|  */ | ||||
|  | ||||
| #include "fastcgi.h" | ||||
| @@ -15,6 +16,7 @@ | ||||
| #include <errno.h> | ||||
| #include <string.h> | ||||
|  | ||||
|  | ||||
| char *fastcgi_add_param(char *buf, const char *key, const char *value) { | ||||
|     char *ptr = buf; | ||||
|     unsigned long key_len = strlen(key); | ||||
| @@ -72,8 +74,8 @@ int fastcgi_init(fastcgi_conn *conn, int mode, unsigned int client_num, unsigned | ||||
|     conn->socket = fcgi_sock; | ||||
|  | ||||
|     struct sockaddr_un sock_addr = {AF_UNIX}; | ||||
|     if (conn->mode == FASTCGI_NECRONDA) { | ||||
|         snprintf(sock_addr.sun_path, sizeof(sock_addr.sun_path) - 1, "%s", NECRONDA_BACKEND_SOCKET); | ||||
|     if (conn->mode == FASTCGI_SESIMOS) { | ||||
|         snprintf(sock_addr.sun_path, sizeof(sock_addr.sun_path) - 1, "%s", SESIMOS_BACKEND_SOCKET); | ||||
|     } else if (conn->mode == FASTCGI_PHP) { | ||||
|         snprintf(sock_addr.sun_path, sizeof(sock_addr.sun_path) - 1, "%s", PHP_FPM_SOCKET); | ||||
|     } | ||||
| @@ -343,7 +345,7 @@ int fastcgi_header(fastcgi_conn *conn, http_res *res, char *err_msg) { | ||||
|             free(content); | ||||
|             return 1; | ||||
|         } else if (header.type == FCGI_STDERR) { | ||||
|             // TODO implement Necronda backend error handling | ||||
|             // TODO implement Sesimos backend error handling | ||||
|             if (conn->mode == FASTCGI_PHP) { | ||||
|                 err = err || fastcgi_php_error(conn, content, content_len, err_msg); | ||||
|             } | ||||
| @@ -386,7 +388,7 @@ int fastcgi_header(fastcgi_conn *conn, http_res *res, char *err_msg) { | ||||
|             return 1; | ||||
|         } | ||||
|  | ||||
|         ret = http_parse_header_field(&res->hdr, ptr, pos0); | ||||
|         ret = http_parse_header_field(&res->hdr, ptr, pos0, 0); | ||||
|         if (ret != 0) return (int) ret; | ||||
|         if (pos0[2] == '\r' && pos0[3] == '\n') { | ||||
|             return 0; | ||||
| @@ -484,7 +486,7 @@ int fastcgi_send(fastcgi_conn *conn, sock *client, int flags) { | ||||
|  | ||||
|             return 0; | ||||
|         } else if (header.type == FCGI_STDERR) { | ||||
|             // TODO implement Necronda backend error handling | ||||
|             // TODO implement Sesimos backend error handling | ||||
|             if (conn->mode == FASTCGI_PHP) { | ||||
|                 fastcgi_php_error(conn, content, content_len, buf0); | ||||
|             } | ||||
| @@ -570,7 +572,7 @@ int fastcgi_dump(fastcgi_conn *conn, char *buf, long len) { | ||||
|  | ||||
|             return 0; | ||||
|         } else if (header.type == FCGI_STDERR) { | ||||
|             // TODO implement Necronda backend error handling | ||||
|             // TODO implement Sesimos backend error handling | ||||
|             if (conn->mode == FASTCGI_PHP) { | ||||
|                 fastcgi_php_error(conn, content, content_len, buf0); | ||||
|             } | ||||
| @@ -598,12 +600,6 @@ int fastcgi_receive(fastcgi_conn *conn, sock *client, unsigned long len) { | ||||
|             .reserved = 0 | ||||
|     }; | ||||
|  | ||||
|     if (client->buf != NULL && client->buf_len - client->buf_off > 0) { | ||||
|         ret = (int) (client->buf_len - client->buf_off); | ||||
|         memcpy(buf, client->buf + client->buf_off, ret); | ||||
|         goto send; | ||||
|     } | ||||
|  | ||||
|     while (rcv_len < len) { | ||||
|         ret = sock_recv(client, buf, sizeof(buf), 0); | ||||
|         if (ret <= 0) { | ||||
| @@ -611,7 +607,6 @@ int fastcgi_receive(fastcgi_conn *conn, sock *client, unsigned long len) { | ||||
|             return -1; | ||||
|         } | ||||
|  | ||||
|         send: | ||||
|         rcv_len += ret; | ||||
|         header.contentLengthB1 = (ret >> 8) & 0xFF; | ||||
|         header.contentLengthB0 = ret & 0xFF; | ||||
| @@ -624,3 +619,21 @@ int fastcgi_receive(fastcgi_conn *conn, sock *client, unsigned long len) { | ||||
|     } | ||||
|     return 0; | ||||
| } | ||||
|  | ||||
| int fastcgi_receive_chunked(fastcgi_conn *conn, sock *client) { | ||||
|     long ret; | ||||
|     unsigned long next_len; | ||||
|  | ||||
|     while (1) { | ||||
|         ret = sock_get_chunk_header(client); | ||||
|         if (ret < 0) return (int) ret; | ||||
|  | ||||
|         next_len = ret; | ||||
|         if (next_len <= 0) break; | ||||
|  | ||||
|         ret = fastcgi_receive(conn, client, next_len); | ||||
|         if (ret < 0) return (int) ret; | ||||
|     } | ||||
|  | ||||
|     return 0; | ||||
| } | ||||
|   | ||||
| @@ -1,12 +1,13 @@ | ||||
| /** | ||||
|  * Necronda Web Server | ||||
|  * FastCGI interface implementation (header file) | ||||
|  * src/lib/fastcgi.h | ||||
|  * Lorenz Stechauner, 2020-12-26 | ||||
|  * sesimos - secure, simple, modern web server | ||||
|  * @brief FastCGI interface implementation (header file) | ||||
|  * @file src/lib/fastcgi.h | ||||
|  * @author Lorenz Stechauner | ||||
|  * @date 2020-12-26 | ||||
|  */ | ||||
|  | ||||
| #ifndef NECRONDA_SERVER_FASTCGI_H | ||||
| #define NECRONDA_SERVER_FASTCGI_H | ||||
| #ifndef SESIMOS_FASTCGI_H | ||||
| #define SESIMOS_FASTCGI_H | ||||
|  | ||||
| #include "include/fastcgi.h" | ||||
| #include "http.h" | ||||
| @@ -19,13 +20,13 @@ | ||||
| #define FASTCGI_COMPRESS_HOLD 8 | ||||
|  | ||||
| #define FASTCGI_PHP 1 | ||||
| #define FASTCGI_NECRONDA 2 | ||||
| #define FASTCGI_SESIMOS 2 | ||||
|  | ||||
| #ifndef PHP_FPM_SOCKET | ||||
| #   define PHP_FPM_SOCKET "/var/run/php-fpm/php-fpm.sock" | ||||
| #endif | ||||
|  | ||||
| #define NECRONDA_BACKEND_SOCKET "/var/run/necronda/necronda-backend.sock" | ||||
| #define SESIMOS_BACKEND_SOCKET "/var/run/sesimos/backend.sock" | ||||
|  | ||||
| typedef struct { | ||||
|     int mode; | ||||
| @@ -54,4 +55,6 @@ int fastcgi_dump(fastcgi_conn *conn, char *buf, long len); | ||||
|  | ||||
| int fastcgi_receive(fastcgi_conn *conn, sock *client, unsigned long len); | ||||
|  | ||||
| #endif //NECRONDA_SERVER_FASTCGI_H | ||||
| int fastcgi_receive_chunked(fastcgi_conn *conn, sock *client); | ||||
|  | ||||
| #endif //SESIMOS_FASTCGI_H | ||||
|   | ||||
| @@ -1,12 +1,14 @@ | ||||
| /** | ||||
|  * Necronda Web Server | ||||
|  * MaxMind GeoIP Database interface | ||||
|  * src/lib/geoip.c | ||||
|  * Lorenz Stechauner, 2021-05-04 | ||||
|  * sesimos - secure, simple, modern web server | ||||
|  * @brief MaxMind GeoIP Database interface | ||||
|  * @file src/lib/geoip.c | ||||
|  * @author Lorenz Stechauner | ||||
|  * @date 2021-05-04 | ||||
|  */ | ||||
|  | ||||
| #include "geoip.h" | ||||
|  | ||||
|  | ||||
| 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: | ||||
|   | ||||
| @@ -1,13 +1,14 @@ | ||||
| /** | ||||
|  * Necronda Web Server | ||||
|  * MaxMind GeoIP Database interface (header file) | ||||
|  * src/lib/geoip.h | ||||
|  * Lorenz Stechauner, 2021-05-04 | ||||
|  * sesimos - secure, simple, modern web server | ||||
|  * @brief MaxMind GeoIP Database interface (header file) | ||||
|  * @file src/lib/geoip.h | ||||
|  * @author Lorenz Stechauner | ||||
|  * @date 2021-05-04 | ||||
|  */ | ||||
|  | ||||
|  | ||||
| #ifndef NECRONDA_SERVER_GEOIP_H | ||||
| #define NECRONDA_SERVER_GEOIP_H | ||||
| #ifndef SESIMOS_GEOIP_H | ||||
| #define SESIMOS_GEOIP_H | ||||
|  | ||||
| #include <maxminddb.h> | ||||
|  | ||||
| @@ -15,4 +16,4 @@ | ||||
|  | ||||
| MMDB_entry_data_list_s *mmdb_json(MMDB_entry_data_list_s *list, char *str, long *str_off, long str_len); | ||||
|  | ||||
| #endif //NECRONDA_SERVER_GEOIP_H | ||||
| #endif //SESIMOS_GEOIP_H | ||||
|   | ||||
| @@ -1,8 +1,9 @@ | ||||
| /** | ||||
|  * Necronda Web Server | ||||
|  * HTTP implementation | ||||
|  * src/lib/http.c | ||||
|  * Lorenz Stechauner, 2020-12-09 | ||||
|  * sesimos - secure, simple, modern web server | ||||
|  * @brief HTTP implementation | ||||
|  * @file src/lib/http.c | ||||
|  * @author Lorenz Stechauner | ||||
|  * @date 2020-12-09 | ||||
|  */ | ||||
|  | ||||
| #include "http.h" | ||||
| @@ -11,6 +12,7 @@ | ||||
|  | ||||
| #include <string.h> | ||||
|  | ||||
|  | ||||
| void http_to_camel_case(char *str, int mode) { | ||||
|     if (mode == HTTP_PRESERVE) | ||||
|         return; | ||||
| @@ -83,7 +85,7 @@ void http_free_res(http_res *res) { | ||||
|     http_free_hdr(&res->hdr); | ||||
| } | ||||
|  | ||||
| int http_parse_header_field(http_hdr *hdr, const char *buf, const char *end_ptr) { | ||||
| int http_parse_header_field(http_hdr *hdr, const char *buf, const char *end_ptr, int flags) { | ||||
|     if (hdr->last_field_num > hdr->field_num) { | ||||
|         print(ERR_STR "Unable to parse header: Invalid state" CLR_STR); | ||||
|         return 3; | ||||
| @@ -116,7 +118,7 @@ int http_parse_header_field(http_hdr *hdr, const char *buf, const char *end_ptr) | ||||
|  | ||||
|     char field_num = hdr->field_num; | ||||
|     int found = http_get_header_field_num_len(hdr, buf, len1); | ||||
|     if (found == -1) { | ||||
|     if (!(flags & HTTP_MERGE_FIELDS) || found == -1) { | ||||
|         if (http_add_header_field_len(hdr, buf, len1, pos1, len2 < 0 ? 0 : len2) != 0) { | ||||
|             print(ERR_STR "Unable to parse header: Too many header fields" CLR_STR); | ||||
|             return 3; | ||||
| @@ -143,7 +145,7 @@ int http_receive_request(sock *client, http_req *req) { | ||||
|     req->hdr.last_field_num = -1; | ||||
|  | ||||
|     while (1) { | ||||
|         rcv_len  = sock_recv(client, buf, CLIENT_MAX_HEADER_SIZE, 0); | ||||
|         rcv_len = sock_recv(client, buf, CLIENT_MAX_HEADER_SIZE, MSG_PEEK); | ||||
|         if (rcv_len <= 0) { | ||||
|             print("Unable to receive http header: %s", sock_strerror(client)); | ||||
|             return -1; | ||||
| @@ -153,6 +155,8 @@ int http_receive_request(sock *client, http_req *req) { | ||||
|         if (header_len <= 0) { | ||||
|             print(ERR_STR "Unable to parse http header: End of header not found" CLR_STR); | ||||
|             return 5; | ||||
|         } else { | ||||
|             rcv_len = sock_recv(client, buf, header_len, 0); | ||||
|         } | ||||
|  | ||||
|         for (int i = 0; i < header_len; i++) { | ||||
| @@ -204,7 +208,7 @@ int http_receive_request(sock *client, http_req *req) { | ||||
|                 sprintf(req->uri, "%.*s", (int) len, pos1); | ||||
|                 sprintf(req->version, "%.3s", pos2 + 5); | ||||
|             } else { | ||||
|                 int ret = http_parse_header_field(&req->hdr, ptr, pos0); | ||||
|                 int ret = http_parse_header_field(&req->hdr, ptr, pos0, HTTP_MERGE_FIELDS); | ||||
|                 if (ret != 0) return ret; | ||||
|             } | ||||
|             ptr = pos0 + 2; | ||||
| @@ -214,13 +218,6 @@ int http_receive_request(sock *client, http_req *req) { | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     client->buf_len = rcv_len - (pos0 - buf + 4); | ||||
|     if (client->buf_len > 0) { | ||||
|         client->buf = malloc(client->buf_len); | ||||
|         client->buf_off = 0; | ||||
|         memcpy(client->buf, pos0 + 4, client->buf_len); | ||||
|     } | ||||
|  | ||||
|     return 0; | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -1,12 +1,13 @@ | ||||
| /** | ||||
|  * Necronda Web Server | ||||
|  * HTTP implementation (header file) | ||||
|  * src/lib/http.h | ||||
|  * Lorenz Stechauner, 2020-12-09 | ||||
|  * sesimos - secure, simple, modern web server | ||||
|  * @brief HTTP implementation (header file) | ||||
|  * @file src/lib/http.h | ||||
|  * @author Lorenz Stechauner | ||||
|  * @date 2020-12-09 | ||||
|  */ | ||||
|  | ||||
| #ifndef NECRONDA_SERVER_HTTP_H | ||||
| #define NECRONDA_SERVER_HTTP_H | ||||
| #ifndef SESIMOS_HTTP_H | ||||
| #define SESIMOS_HTTP_H | ||||
|  | ||||
| #include "sock.h" | ||||
|  | ||||
| @@ -22,6 +23,8 @@ | ||||
| #define HTTP_FIELD_EX_VALUE 1 | ||||
| #define HTTP_FIELD_EX_NAME 2 | ||||
|  | ||||
| #define HTTP_MERGE_FIELDS 1 | ||||
|  | ||||
| #define HTTP_1XX_STR "\x1B[1;32m" | ||||
| #define HTTP_2XX_STR "\x1B[1;32m" | ||||
| #define HTTP_3XX_STR "\x1B[1;33m" | ||||
| @@ -37,11 +40,11 @@ | ||||
| #define HTTP_MAX_HEADER_FIELD_NUM 64 | ||||
|  | ||||
| #ifndef SERVER_STR | ||||
| #   define SERVER_STR "Necronda" | ||||
| #   define SERVER_STR "Sesimos" | ||||
| #endif | ||||
|  | ||||
| #ifndef SERVER_STR_HTML | ||||
| #   define SERVER_STR_HTML "Necronda web server" | ||||
| #   define SERVER_STR_HTML "Sesimos web server" | ||||
| #endif | ||||
|  | ||||
| typedef struct { | ||||
| @@ -106,6 +109,7 @@ typedef enum { | ||||
| typedef struct { | ||||
|     unsigned short status; | ||||
|     http_error_origin origin; | ||||
|     const char* ws_key; | ||||
| } http_status_ctx; | ||||
|  | ||||
| extern const http_status http_statuses[]; | ||||
| @@ -140,7 +144,7 @@ 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) ; | ||||
| int http_parse_header_field(http_hdr *hdr, const char *buf, const char *end_ptr, int flags); | ||||
|  | ||||
| const char *http_get_header_field(const http_hdr *hdr, const char *field_name); | ||||
|  | ||||
| @@ -176,4 +180,4 @@ const http_doc_info *http_get_status_info(const http_status *status); | ||||
|  | ||||
| int http_get_compression(const http_req *req, const http_res *res); | ||||
|  | ||||
| #endif //NECRONDA_SERVER_HTTP_H | ||||
| #endif //SESIMOS_HTTP_H | ||||
|   | ||||
| @@ -1,13 +1,15 @@ | ||||
| /** | ||||
|  * Necronda Web Server | ||||
|  * HTTP static implementation | ||||
|  * src/lib/http_static.c | ||||
|  * Lorenz Stechauner, 2021-05-03 | ||||
|  * sesimos - secure, simple, modern web server | ||||
|  * @brief HTTP static implementation | ||||
|  * @file src/lib/http_static.c | ||||
|  * @author Lorenz Stechauner | ||||
|  * @date 2021-05-03 | ||||
|  */ | ||||
|  | ||||
| #include "../necronda.h" | ||||
| #include "../defs.h" | ||||
| #include "http.h" | ||||
|  | ||||
|  | ||||
| const http_status http_statuses[] = { | ||||
|         {100, "Informational", "Continue"}, | ||||
|         {101, "Informational", "Switching Protocols"}, | ||||
|   | ||||
| @@ -1,12 +1,11 @@ | ||||
| /** | ||||
|  * Necronda Web Server | ||||
|  * FastCGI header file | ||||
|  * src/lib/include/fastcgi.h | ||||
|  * Lorenz Stechauner, 2021-05-03 | ||||
|  * sesimos - secure, simple, modern web server | ||||
|  * @brief FastCGI header file | ||||
|  * @file src/lib/include/fastcgi.h | ||||
|  */ | ||||
|  | ||||
| #ifndef NECRONDA_SERVER_EXTERN_FASTCGI_H | ||||
| #define NECRONDA_SERVER_EXTERN_FASTCGI_H | ||||
| #ifndef SESIMOS_EXTERN_FASTCGI_H | ||||
| #define SESIMOS_EXTERN_FASTCGI_H | ||||
|  | ||||
| /* | ||||
|  * Listening socket file number | ||||
| @@ -119,4 +118,4 @@ typedef struct { | ||||
|     FCGI_UnknownTypeBody body; | ||||
| } FCGI_UnknownTypeRecord; | ||||
|  | ||||
| #endif //NECRONDA_SERVER_EXTERN_FASTCGI_H | ||||
| #endif //SESIMOS_EXTERN_FASTCGI_H | ||||
|   | ||||
| @@ -1,14 +1,16 @@ | ||||
| /** | ||||
|  * Necronda Web Server | ||||
|  * Reverse proxy | ||||
|  * src/lib/rev_proxy.c | ||||
|  * Lorenz Stechauner, 2021-01-07 | ||||
|  * sesimos - secure, simple, modern web server | ||||
|  * @brief Reverse proxy | ||||
|  * @file src/lib/rev_proxy.c | ||||
|  * @author Lorenz Stechauner | ||||
|  * @date 2021-01-07 | ||||
|  */ | ||||
|  | ||||
| #include "../defs.h" | ||||
| #include "../server.h" | ||||
| #include "rev_proxy.h" | ||||
| #include "utils.h" | ||||
| #include "compress.h" | ||||
| #include "../server.h" | ||||
|  | ||||
| #include <openssl/ssl.h> | ||||
| #include <string.h> | ||||
| @@ -17,14 +19,12 @@ | ||||
| #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; | ||||
| int rev_proxy_preload(void) { | ||||
|     rev_proxy.ctx = SSL_CTX_new(TLS_client_method()); | ||||
|     return 0; | ||||
| } | ||||
| @@ -32,8 +32,6 @@ int rev_proxy_preload() { | ||||
| int rev_proxy_request_header(http_req *req, int enc) { | ||||
|     char buf1[256], buf2[256]; | ||||
|     int p_len; | ||||
|     http_remove_header_field(&req->hdr, "Connection", HTTP_REMOVE_ALL); | ||||
|     http_add_header_field(&req->hdr, "Connection", "keep-alive"); | ||||
|  | ||||
|     const char *via = http_get_header_field(&req->hdr, "Via"); | ||||
|     sprintf(buf1, "HTTP/%s %s", req->version, SERVER_NAME); | ||||
| @@ -183,12 +181,12 @@ int rev_proxy_response_header(http_req *req, http_res *res, host_config *conf) { | ||||
|  | ||||
| int rev_proxy_init(http_req *req, http_res *res, http_status_ctx *ctx, host_config *conf, sock *client, http_status *custom_status, char *err_msg) { | ||||
|     char buffer[CHUNK_SIZE]; | ||||
|     const char *connection, *upgrade, *ws_version; | ||||
|     long ret; | ||||
|     int tries = 0, retry = 0; | ||||
|  | ||||
|     if (rev_proxy.socket != 0 && strcmp(rev_proxy_host, conf->name) == 0 && sock_check(&rev_proxy) == 0) { | ||||
|     if (rev_proxy.socket != 0 && strcmp(rev_proxy_host, conf->name) == 0 && sock_check(&rev_proxy) == 0) | ||||
|         goto rev_proxy; | ||||
|     } | ||||
|  | ||||
|     retry: | ||||
|     if (rev_proxy.socket != 0) { | ||||
| @@ -289,6 +287,22 @@ int rev_proxy_init(http_req *req, http_res *res, http_status_ctx *ctx, host_conf | ||||
|     print(BLUE_STR "Established new connection with " BLD_STR "[%s]:%i" CLR_STR, buffer, conf->rev_proxy.port); | ||||
|  | ||||
|     rev_proxy: | ||||
|     connection = http_get_header_field(&req->hdr, "Connection"); | ||||
|     if (connection != NULL && (strstr(connection, "upgrade") != NULL || strstr(connection, "Upgrade") != NULL)) { | ||||
|         upgrade = http_get_header_field(&req->hdr, "Upgrade"); | ||||
|         ws_version = http_get_header_field(&req->hdr, "Sec-WebSocket-Version"); | ||||
|         if (upgrade != NULL && ws_version != NULL && strcmp(upgrade, "websocket") == 0 && strcmp(ws_version, "13") == 0) { | ||||
|             ctx->ws_key = http_get_header_field(&req->hdr, "Sec-WebSocket-Key"); | ||||
|         } else { | ||||
|             res->status = http_get_status(501); | ||||
|             ctx->origin = INTERNAL; | ||||
|             return -1; | ||||
|         } | ||||
|     } else { | ||||
|         http_remove_header_field(&req->hdr, "Connection", HTTP_REMOVE_ALL); | ||||
|         http_add_header_field(&req->hdr, "Connection", "keep-alive"); | ||||
|     } | ||||
|  | ||||
|     ret = rev_proxy_request_header(req, (int) client->enc); | ||||
|     if (ret != 0) { | ||||
|         res->status = http_get_status(500); | ||||
| @@ -307,46 +321,35 @@ int rev_proxy_init(http_req *req, http_res *res, http_status_ctx *ctx, host_conf | ||||
|     } | ||||
|  | ||||
|     const char *content_length = http_get_header_field(&req->hdr, "Content-Length"); | ||||
|     if (content_length != NULL) { | ||||
|         unsigned long content_len = strtoul(content_length, NULL, 10); | ||||
|         if (client->buf_len - client->buf_off > 0) { | ||||
|             unsigned long len = client->buf_len - client->buf_off; | ||||
|             if (len > content_len) { | ||||
|                 len = content_len; | ||||
|             } | ||||
|             ret = sock_send(&rev_proxy, client->buf, len, 0); | ||||
|             if (ret <= 0) { | ||||
|                 res->status = http_get_status(502); | ||||
|                 ctx->origin = SERVER_REQ; | ||||
|                 print(ERR_STR "Unable to send request to server (2): %s" CLR_STR, sock_strerror(&rev_proxy)); | ||||
|                 sprintf(err_msg, "Unable to send request to server: %s.", sock_strerror(&rev_proxy)); | ||||
|                 retry = tries < 4; | ||||
|                 goto proxy_err; | ||||
|             } | ||||
|             content_len -= len; | ||||
|         } | ||||
|         if (content_len > 0) { | ||||
|             ret = sock_splice(&rev_proxy, client, buffer, sizeof(buffer), content_len); | ||||
|             if (ret <= 0) { | ||||
|                 if (ret == -1) { | ||||
|                     res->status = http_get_status(502); | ||||
|                     ctx->origin = SERVER_REQ; | ||||
|                     print(ERR_STR "Unable to send request to server (3): %s" CLR_STR, sock_strerror(&rev_proxy)); | ||||
|                     sprintf(err_msg, "Unable to send request to server: %s.", sock_strerror(&rev_proxy)); | ||||
|                     goto proxy_err; | ||||
|                 } else if (ret == -2) { | ||||
|                     res->status = http_get_status(400); | ||||
|                     ctx->origin = CLIENT_REQ; | ||||
|                     print(ERR_STR "Unable to receive request from client: %s" CLR_STR, sock_strerror(client)); | ||||
|                     sprintf(err_msg, "Unable to receive request from client: %s.", sock_strerror(client)); | ||||
|                     return -1; | ||||
|                 } | ||||
|                 res->status = http_get_status(500); | ||||
|                 ctx->origin = INTERNAL; | ||||
|                 print(ERR_STR "Unknown Error" CLR_STR); | ||||
|                 return -1; | ||||
|             } | ||||
|     unsigned long content_len = content_length != NULL ? strtoul(content_length, NULL, 10) : 0; | ||||
|     const char *transfer_encoding = http_get_header_field(&req->hdr, "Transfer-Encoding"); | ||||
|  | ||||
|     ret = 0; | ||||
|     if (content_len > 0) { | ||||
|         ret = sock_splice(&rev_proxy, client, buffer, sizeof(buffer), content_len); | ||||
|     } else if (transfer_encoding != NULL && strstr(transfer_encoding, "chunked") != NULL) { | ||||
|         ret = sock_splice_chunked(&rev_proxy, client, buffer, sizeof(buffer)); | ||||
|     } | ||||
|  | ||||
|     if (ret < 0 || (content_len != 0 && ret != content_len)) { | ||||
|         if (ret == -1) { | ||||
|             res->status = http_get_status(502); | ||||
|             ctx->origin = SERVER_REQ; | ||||
|             print(ERR_STR "Unable to send request to server (2): %s" CLR_STR, sock_strerror(&rev_proxy)); | ||||
|             sprintf(err_msg, "Unable to send request to server: %s.", sock_strerror(&rev_proxy)); | ||||
|             retry = tries < 4; | ||||
|             goto proxy_err; | ||||
|         } else if (ret == -2) { | ||||
|             res->status = http_get_status(400); | ||||
|             ctx->origin = CLIENT_REQ; | ||||
|             print(ERR_STR "Unable to receive request from client: %s" CLR_STR, sock_strerror(client)); | ||||
|             sprintf(err_msg, "Unable to receive request from client: %s.", sock_strerror(client)); | ||||
|             return -1; | ||||
|         } | ||||
|         res->status = http_get_status(500); | ||||
|         ctx->origin = INTERNAL; | ||||
|         print(ERR_STR "Unknown Error" CLR_STR); | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     ret = sock_recv(&rev_proxy, buffer, sizeof(buffer), MSG_PEEK); | ||||
| @@ -422,7 +425,7 @@ int rev_proxy_init(http_req *req, http_res *res, http_status_ctx *ctx, host_conf | ||||
|                 goto proxy_err; | ||||
|             } | ||||
|         } else { | ||||
|             ret = http_parse_header_field(&res->hdr, ptr, pos0); | ||||
|             ret = http_parse_header_field(&res->hdr, ptr, pos0, 0); | ||||
|             if (ret != 0) { | ||||
|                 res->status = http_get_status(502); | ||||
|                 ctx->origin = SERVER_RES; | ||||
| @@ -453,7 +456,6 @@ int rev_proxy_init(http_req *req, http_res *res, http_status_ctx *ctx, host_conf | ||||
| } | ||||
|  | ||||
| int rev_proxy_send(sock *client, unsigned long len_to_send, int flags) { | ||||
|     // TODO handle websockets | ||||
|     char buffer[CHUNK_SIZE], comp_out[CHUNK_SIZE], buf[256], *ptr; | ||||
|     long ret = 0, len, snd_len; | ||||
|     int finish_comp = 0; | ||||
| @@ -476,20 +478,18 @@ int rev_proxy_send(sock *client, unsigned long len_to_send, int flags) { | ||||
|     do { | ||||
|         snd_len = 0; | ||||
|         if (flags & REV_PROXY_CHUNKED) { | ||||
|             char *pos; | ||||
|             ret = sock_recv(&rev_proxy, buffer, 16, MSG_PEEK); | ||||
|             if (ret <= 0) goto err0; | ||||
|  | ||||
|             len_to_send = strtol(buffer, NULL, 16); | ||||
|             pos = strstr(buffer, "\r\n"); | ||||
|             len = pos - buffer + 2; | ||||
|             sock_recv(&rev_proxy, buffer, len, 0); | ||||
|             if (ret <= 0) { | ||||
|                 err0: | ||||
|                 print("Unable to receive from server: %s", sock_strerror(&rev_proxy)); | ||||
|             ret = sock_get_chunk_header(&rev_proxy); | ||||
|             if (ret < 0) { | ||||
|                 if (ret == -1) { | ||||
|                     print("Unable to receive from server: Malformed chunk header"); | ||||
|                 } else { | ||||
|                     print("Unable to receive from server: %s", sock_strerror(&rev_proxy)); | ||||
|                 } | ||||
|                 break; | ||||
|             } | ||||
|  | ||||
|             len_to_send = ret; | ||||
|             ret = 1; | ||||
|             if (len_to_send == 0 && (flags & REV_PROXY_COMPRESS)) { | ||||
|                 finish_comp = 1; | ||||
|                 len = 0; | ||||
| @@ -523,11 +523,14 @@ int rev_proxy_send(sock *client, unsigned long len_to_send, int flags) { | ||||
|                 if (buf_len != 0) { | ||||
|                     len = sprintf(buf, "%lX\r\n", buf_len); | ||||
|                     ret = 1; | ||||
|  | ||||
|                     if (flags & REV_PROXY_CHUNKED) ret = sock_send(client, buf, len, 0); | ||||
|                     if (ret <= 0) goto err; | ||||
|  | ||||
|                     ret = sock_send(client, ptr, buf_len, 0); | ||||
|                     if (ret <= 0) goto err; | ||||
|                     if (!(flags & REV_PROXY_COMPRESS)) snd_len += ret; | ||||
|  | ||||
|                     if (flags & REV_PROXY_CHUNKED) ret = sock_send(client, "\r\n", 2, 0); | ||||
|                     if (ret <= 0) { | ||||
|                         err: | ||||
| @@ -543,7 +546,7 @@ int rev_proxy_send(sock *client, unsigned long len_to_send, int flags) { | ||||
|         if (flags & REV_PROXY_CHUNKED) sock_recv(&rev_proxy, buffer, 2, 0); | ||||
|     } while ((flags & REV_PROXY_CHUNKED) && len_to_send > 0); | ||||
|  | ||||
|     if (ret <= 0) return (int) -1; | ||||
|     if (ret <= 0) return -1; | ||||
|  | ||||
|     if (flags & REV_PROXY_CHUNKED) { | ||||
|         ret = sock_send(client, "0\r\n\r\n", 5, 0); | ||||
|   | ||||
| @@ -1,12 +1,13 @@ | ||||
| /** | ||||
|  * Necronda Web Server | ||||
|  * Reverse proxy (header file) | ||||
|  * src/lib/rev_proxy.h | ||||
|  * Lorenz Stechauner, 2021-01-07 | ||||
|  * sesimos - secure, simple, modern web server | ||||
|  * @brief Reverse proxy (header file) | ||||
|  * @file src/lib/rev_proxy.h | ||||
|  * @author Lorenz Stechauner | ||||
|  * @date 2021-01-07 | ||||
|  */ | ||||
|  | ||||
| #ifndef NECRONDA_SERVER_REV_PROXY_H | ||||
| #define NECRONDA_SERVER_REV_PROXY_H | ||||
| #ifndef SESIMOS_REV_PROXY_H | ||||
| #define SESIMOS_REV_PROXY_H | ||||
|  | ||||
| #define REV_PROXY_CHUNKED 1 | ||||
| #define REV_PROXY_COMPRESS_GZ 2 | ||||
| @@ -22,7 +23,7 @@ | ||||
|  | ||||
| extern sock rev_proxy; | ||||
|  | ||||
| int rev_proxy_preload(); | ||||
| int rev_proxy_preload(void); | ||||
|  | ||||
| int rev_proxy_request_header(http_req *req, int enc); | ||||
|  | ||||
| @@ -35,4 +36,4 @@ int rev_proxy_send(sock *client, unsigned long len_to_send, int flags); | ||||
|  | ||||
| int rev_proxy_dump(char *buf, long len); | ||||
|  | ||||
| #endif //NECRONDA_SERVER_REV_PROXY_H | ||||
| #endif //SESIMOS_REV_PROXY_H | ||||
|   | ||||
							
								
								
									
										105
									
								
								src/lib/sock.c
									
									
									
									
									
								
							
							
						
						
									
										105
									
								
								src/lib/sock.c
									
									
									
									
									
								
							| @@ -1,17 +1,21 @@ | ||||
| /** | ||||
|  * Necronda Web Server | ||||
|  * Basic TCP and TLS socket | ||||
|  * src/lib/sock.c | ||||
|  * Lorenz Stechauner, 2021-01-07 | ||||
|  * sesimos - secure, simple, modern web server | ||||
|  * @brief Basic TCP and TLS socket | ||||
|  * @file src/lib/sock.c | ||||
|  * @author Lorenz Stechauner | ||||
|  * @date 2021-01-07 | ||||
|  */ | ||||
|  | ||||
| #include "sock.h" | ||||
| #include "utils.h" | ||||
|  | ||||
| #include <openssl/err.h> | ||||
| #include <openssl/ssl.h> | ||||
| #include <string.h> | ||||
| #include <sys/socket.h> | ||||
| #include <unistd.h> | ||||
| #include <poll.h> | ||||
|  | ||||
|  | ||||
| int sock_enc_error(sock *s) { | ||||
|     return (int) s->enc ? SSL_get_error(s->ssl, (int) s->_last_ret) : 0; | ||||
| @@ -69,11 +73,8 @@ long sock_send(sock *s, void *buf, unsigned long len, int flags) { | ||||
| long sock_recv(sock *s, void *buf, unsigned long len, int flags) { | ||||
|     long ret; | ||||
|     if (s->enc) { | ||||
|         if (flags & MSG_PEEK) { | ||||
|             ret = SSL_peek(s->ssl, buf, (int) len); | ||||
|         } else { | ||||
|             ret = SSL_read(s->ssl, buf, (int) len); | ||||
|         } | ||||
|         int (*func)(SSL*, void*, int) = (flags & MSG_PEEK) ? SSL_peek : SSL_read; | ||||
|         ret = func(s->ssl, buf, (int) len); | ||||
|         s->_ssl_error = ERR_get_error(); | ||||
|     } else { | ||||
|         ret = recv(s->socket, buf, len, flags); | ||||
| @@ -90,7 +91,7 @@ long sock_splice(sock *dst, sock *src, void *buf, unsigned long buf_len, unsigne | ||||
|     while (send_len < len) { | ||||
|         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 <= 0) return -2; | ||||
|         next_len = ret; | ||||
|         ret = sock_send(dst, buf, next_len, send_len + next_len < len ? MSG_MORE : 0); | ||||
|         if (ret < 0) return -1; | ||||
| @@ -100,6 +101,25 @@ long sock_splice(sock *dst, sock *src, void *buf, unsigned long buf_len, unsigne | ||||
|     return (long) send_len; | ||||
| } | ||||
|  | ||||
| long sock_splice_chunked(sock *dst, sock *src, void *buf, unsigned long buf_len) { | ||||
|     long ret; | ||||
|     unsigned long send_len = 0; | ||||
|     unsigned long next_len; | ||||
|  | ||||
|     while (1) { | ||||
|         ret = sock_get_chunk_header(src); | ||||
|         if (ret < 0) return -2; | ||||
|  | ||||
|         next_len = ret; | ||||
|         if (next_len <= 0) break; | ||||
|  | ||||
|         ret = sock_splice(dst, src, buf, buf_len, next_len); | ||||
|         if (ret < 0) return ret; | ||||
|     } | ||||
|  | ||||
|     return (long) send_len; | ||||
| } | ||||
|  | ||||
| int sock_close(sock *s) { | ||||
|     if ((int) s->enc && s->ssl != NULL) { | ||||
|         if (s->_last_ret >= 0) SSL_shutdown(s->ssl); | ||||
| @@ -117,3 +137,68 @@ int sock_check(sock *s) { | ||||
|     char buf; | ||||
|     return recv(s->socket, &buf, 1, MSG_PEEK | MSG_DONTWAIT) == 1; | ||||
| } | ||||
|  | ||||
| int sock_poll(sock *sockets[], sock *ready[], sock *error[], int n_sock, int *n_ready, int *n_error, short events, int timeout_ms) { | ||||
|     struct pollfd fds[n_sock]; | ||||
|     for (int i = 0; i < n_sock; i++) { | ||||
|         fds[i].fd = sockets[i]->socket; | ||||
|         fds[i].events = events; | ||||
|     } | ||||
|  | ||||
|     int ret = poll(fds, n_sock, timeout_ms); | ||||
|     if (ret < 0 || ready == NULL || error == NULL) return ret; | ||||
|  | ||||
|     *n_ready = 0, *n_error = 0; | ||||
|     for (int i = 0; i < n_sock; i++) { | ||||
|         if (fds[i].revents & events) | ||||
|             ready[(*n_ready)++] = sockets[i]; | ||||
|         if (fds[i].revents & (POLLERR | POLLHUP | POLLNVAL)) | ||||
|             error[(*n_error)++] = sockets[i]; | ||||
|     } | ||||
|  | ||||
|     return ret; | ||||
| } | ||||
|  | ||||
| int sock_poll_read(sock *sockets[], sock *readable[], sock *error[], int n_sock, int *n_readable, int *n_error, int timeout_ms) { | ||||
|     return sock_poll(sockets, readable, error, n_sock, n_readable, n_error, POLLIN, timeout_ms); | ||||
| } | ||||
|  | ||||
| int sock_poll_write(sock *sockets[], sock *writable[], sock *error[], int n_sock, int *n_writable, int *n_error, int timeout_ms) { | ||||
|     return sock_poll(sockets, writable, error, n_sock, n_writable, n_error, POLLOUT, timeout_ms); | ||||
| } | ||||
|  | ||||
| long sock_parse_chunk_header(const char *buf, long len, long *ret_len) { | ||||
|     for (int i = 0; i < len; i++) { | ||||
|         char ch = buf[i]; | ||||
|         if (ch == '\r') { | ||||
|             continue; | ||||
|         } else if (ch == '\n') { | ||||
|             if (ret_len != NULL) *ret_len = i + 1; | ||||
|             return strtol(buf, NULL, 16); | ||||
|         } else if (!((ch >= '0' && ch <= '9') || (ch >= 'a' && ch <= 'f') || (ch >= 'A' && ch <= 'F'))) { | ||||
|             return -2; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return -1; | ||||
| } | ||||
|  | ||||
| long sock_get_chunk_header(sock *s) { | ||||
|     long ret, len; | ||||
|     char buf[16]; | ||||
|  | ||||
|     do { | ||||
|         print("debug1");  // TODO remove | ||||
|         ret = sock_recv(s, buf, sizeof(buf), MSG_PEEK); | ||||
|         if (ret <= 0) return -2; | ||||
|         else if (ret < 2) continue; | ||||
|  | ||||
|         ret = sock_parse_chunk_header(buf, ret, &len); | ||||
|         if (ret == -2) return -1; | ||||
|     } while (ret < 0); | ||||
|  | ||||
|     if (sock_recv(s, buf, len, 0) != len) | ||||
|         return -2; | ||||
|  | ||||
|     return ret; | ||||
| } | ||||
|   | ||||
| @@ -1,23 +1,22 @@ | ||||
| /** | ||||
|  * Necronda Web Server | ||||
|  * Basic TCP and TLS socket (header file) | ||||
|  * src/lib/sock.h | ||||
|  * Lorenz Stechauner, 2021-01-07 | ||||
|  * sesimos - secure, simple, modern web server | ||||
|  * @brief Basic TCP and TLS socket (header file) | ||||
|  * @file src/lib/sock.h | ||||
|  * @author Lorenz Stechauner | ||||
|  * @date 2021-01-07 | ||||
|  */ | ||||
|  | ||||
| #ifndef NECRONDA_SERVER_SOCK_H | ||||
| #define NECRONDA_SERVER_SOCK_H | ||||
| #ifndef SESIMOS_SOCK_H | ||||
| #define SESIMOS_SOCK_H | ||||
|  | ||||
| #include <openssl/crypto.h> | ||||
| #include <sys/socket.h> | ||||
|  | ||||
| typedef struct { | ||||
|     unsigned int enc:1; | ||||
|     int socket; | ||||
|     SSL_CTX *ctx; | ||||
|     SSL *ssl; | ||||
|     char *buf; | ||||
|     unsigned long buf_len; | ||||
|     unsigned long buf_off; | ||||
|     long _last_ret; | ||||
|     int _errno; | ||||
|     unsigned long _ssl_error; | ||||
| @@ -33,8 +32,20 @@ long sock_recv(sock *s, void *buf, unsigned long len, int flags); | ||||
|  | ||||
| long sock_splice(sock *dst, sock *src, void *buf, unsigned long buf_len, unsigned long len); | ||||
|  | ||||
| long sock_splice_chunked(sock *dst, sock *src, void *buf, unsigned long buf_len); | ||||
|  | ||||
| int sock_close(sock *s); | ||||
|  | ||||
| int sock_check(sock *s); | ||||
|  | ||||
| #endif //NECRONDA_SERVER_SOCK_H | ||||
| int sock_poll(sock *sockets[], sock *ready[], sock *error[], int n_sock, int *n_ready, int *n_error, short events, int timeout_ms); | ||||
|  | ||||
| int sock_poll_read(sock *sockets[], sock *readable[], sock *error[], int n_sock, int *n_readable, int *n_error, int timeout_ms); | ||||
|  | ||||
| int sock_poll_write(sock *sockets[], sock *writable[], sock *error[], int n_sock, int *n_writable, int *n_error, int timeout_ms); | ||||
|  | ||||
| long sock_parse_chunk_header(const char *buf, long len, long *ret_len); | ||||
|  | ||||
| long sock_get_chunk_header(sock *s); | ||||
|  | ||||
| #endif //SESIMOS_SOCK_H | ||||
|   | ||||
| @@ -1,8 +1,9 @@ | ||||
| /** | ||||
|  * Necronda Web Server | ||||
|  * URI and path handlers | ||||
|  * src/lib/uri.c | ||||
|  * Lorenz Stechauner, 2020-12-13 | ||||
|  * sesimos - secure, simple, modern web server | ||||
|  * @brief URI and path handlers | ||||
|  * @file src/lib/uri.c | ||||
|  * @author Lorenz Stechauner | ||||
|  * @date 2020-12-13 | ||||
|  */ | ||||
|  | ||||
| #include "uri.h" | ||||
| @@ -11,6 +12,7 @@ | ||||
| #include <stdlib.h> | ||||
| #include <string.h> | ||||
|  | ||||
|  | ||||
| int path_is_directory(const char *path) { | ||||
|     struct stat statbuf; | ||||
|     return stat(path, &statbuf) == 0 && S_ISDIR(statbuf.st_mode) != 0; | ||||
|   | ||||
| @@ -1,12 +1,13 @@ | ||||
| /** | ||||
|  * Necronda Web Server | ||||
|  * URI and path handlers (header file) | ||||
|  * src/lib/uri.h | ||||
|  * Lorenz Stechauner, 2020-12-13 | ||||
|  * sesimos - secure, simple, modern web server | ||||
|  * @brief URI and path handlers (header file) | ||||
|  * @file src/lib/uri.h | ||||
|  * @author Lorenz Stechauner | ||||
|  * @date 2020-12-13 | ||||
|  */ | ||||
|  | ||||
| #ifndef NECRONDA_SERVER_URI_H | ||||
| #define NECRONDA_SERVER_URI_H | ||||
| #ifndef SESIMOS_URI_H | ||||
| #define SESIMOS_URI_H | ||||
|  | ||||
| #include <sys/stat.h> | ||||
|  | ||||
| @@ -44,4 +45,4 @@ int uri_init_cache(http_uri *uri); | ||||
|  | ||||
| void uri_free(http_uri *uri); | ||||
|  | ||||
| #endif //NECRONDA_SERVER_URI_H | ||||
| #endif //SESIMOS_URI_H | ||||
|   | ||||
							
								
								
									
										107
									
								
								src/lib/utils.c
									
									
									
									
									
								
							
							
						
						
									
										107
									
								
								src/lib/utils.c
									
									
									
									
									
								
							| @@ -1,8 +1,9 @@ | ||||
| /** | ||||
|  * Necronda Web Server | ||||
|  * Utilities | ||||
|  * src/lib/utils.c | ||||
|  * Lorenz Stechauner, 2020-12-03 | ||||
|  * sesimos - secure, simple, modern web server | ||||
|  * @brief Utilities | ||||
|  * @file src/lib/utils.c | ||||
|  * @author Lorenz Stechauner | ||||
|  * @date 2020-12-03 | ||||
|  */ | ||||
|  | ||||
| #include "utils.h" | ||||
| @@ -11,71 +12,80 @@ | ||||
| #include <string.h> | ||||
| #include <stdlib.h> | ||||
|  | ||||
|  | ||||
| char *log_prefix; | ||||
|  | ||||
| static const char base64_encode_table[64] = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/"; | ||||
| static const int base64_mod_table[3] = {0, 2, 1}; | ||||
|  | ||||
|  | ||||
| char *format_duration(unsigned long micros, char *buf) { | ||||
|     if (micros < 10000) { | ||||
|         sprintf(buf, "%.1f ms", (double) micros / 1000); | ||||
|     } else if (micros < 1000000) { | ||||
|         sprintf(buf, "%li ms", micros / 1000); | ||||
|     } else if (micros < 60000000) { | ||||
|     } else if (micros < 1000000 - 1000) { | ||||
|         sprintf(buf, "%.0f ms", (double) micros / 1000); | ||||
|     } else if (micros < 60000000 - 1000000) { | ||||
|         sprintf(buf, "%.1f s", (double) micros / 1000000); | ||||
|     } else if (micros < 6000000000) { | ||||
|         sprintf(buf, "%.1f min", (double) micros / 1000000 / 60); | ||||
|     } else { | ||||
|         sprintf(buf, "%li min", micros / 1000000 / 60); | ||||
|         sprintf(buf, "%.0f min", (double) micros / 1000000 / 60); | ||||
|     } | ||||
|     return buf; | ||||
| } | ||||
|  | ||||
| int url_encode_component(const char *str, char *enc, long *size) { | ||||
|     char *ptr = enc; | ||||
|     char ch; | ||||
|     memset(enc, 0, *size); | ||||
|     for (int i = 0; i < strlen(str); i++, ptr++) { | ||||
|         if ((ptr - enc) >= *size) { | ||||
|             return -1; | ||||
| int url_encode_component(const void *in, size_t size_in, char *out, size_t size_out) { | ||||
|     int size = 0; | ||||
|  | ||||
|     // Encode control characters | ||||
|     for (int i = 0; i < size_in; i++) { | ||||
|         unsigned char ch = ((unsigned char *) in)[i]; | ||||
|         if (ch == ' ') { | ||||
|             ch = '+'; | ||||
|         } else if ( | ||||
|                 ch <= 0x20 || ch >= 0x7F || | ||||
|                 !((ch >= 'A' && ch <= 'Z') || (ch >= 'a' && ch <= 'z') || (ch >= '0' && ch <= '9') || | ||||
|                   ch == '-' || ch == '_' || ch == '.' || ch == '!' || ch == '~' || ch == '*' || ch == '\'' || | ||||
|                   ch == '(' || ch == ')') | ||||
|         ) { | ||||
|             size += 3; | ||||
|             if (size < size_out) sprintf(out + size - 3, "%%%02X", ch); | ||||
|             ch = 0; | ||||
|         } | ||||
|         ch = str[i]; | ||||
|         if (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; | ||||
|             } | ||||
|             sprintf(ptr, "%%%02X", ch); | ||||
|             ptr += 2; | ||||
|         } else if (ch == ' ') { | ||||
|             ptr[0] = '+'; | ||||
|         } else { | ||||
|             ptr[0] = ch; | ||||
|  | ||||
|         if (ch != 0) { | ||||
|             size++; | ||||
|             if (size < size_out) out[size - 1] = (char) ch; | ||||
|         } | ||||
|     } | ||||
|     *size = ptr - enc; | ||||
|     return 0; | ||||
|  | ||||
|     // Set terminating null byte | ||||
|     if (size_out > 0) out[size < size_out ? size : size_out - 1] = 0; | ||||
|  | ||||
|     // Return theoretical size | ||||
|     return size; | ||||
| } | ||||
|  | ||||
| int url_encode(const char *str, char *enc, long *size) { | ||||
|     char *ptr = enc; | ||||
|     unsigned char ch; | ||||
|     memset(enc, 0, *size); | ||||
|     for (int i = 0; i < strlen(str); i++, ptr++) { | ||||
|         if ((ptr - enc) >= *size) { | ||||
|             return -1; | ||||
|         } | ||||
|         ch = str[i]; | ||||
|         if (ch > 0x7F || ch == ' ') { | ||||
|             if ((ptr - enc + 2) >= *size) { | ||||
|                 return -1; | ||||
|             } | ||||
|             sprintf(ptr, "%%%02X", ch); | ||||
|             ptr += 2; | ||||
| int url_encode(const void *in, size_t size_in, char *out, size_t size_out) { | ||||
|     int size = 0; | ||||
|  | ||||
|     // Encode control characters | ||||
|     for (int i = 0; i < size_in; i++) { | ||||
|         unsigned char ch = ((unsigned char *) in)[i]; | ||||
|         if (ch <= 0x20 || ch >= 0x7F) { | ||||
|             size += 3; | ||||
|             if (size < size_out) sprintf(out + size - 3, "%%%02X", ch); | ||||
|         } else { | ||||
|             ptr[0] = (char) ch; | ||||
|             size++; | ||||
|             if (size < size_out) out[size - 1] = (char) ch; | ||||
|         } | ||||
|     } | ||||
|     *size = ptr - enc; | ||||
|     return 0; | ||||
|  | ||||
|     // Set terminating null byte | ||||
|     if (size_out > 0) out[size < size_out ? size : size_out - 1] = 0; | ||||
|  | ||||
|     // Return theoretical size | ||||
|     return size; | ||||
| } | ||||
|  | ||||
| int url_decode(const char *str, char *dec, long *size) { | ||||
| @@ -191,6 +201,7 @@ int base64_encode(void *data, unsigned long data_len, char *output, unsigned lon | ||||
|  | ||||
|     for (int i = 0; i < base64_mod_table[data_len % 3]; i++) | ||||
|         output[out_len - 1 - i] = '='; | ||||
|     output[out_len] = 0; | ||||
|  | ||||
|     return 0; | ||||
| } | ||||
|   | ||||
| @@ -1,12 +1,13 @@ | ||||
| /** | ||||
|  * Necronda Web Server | ||||
|  * Utilities (header file) | ||||
|  * src/lib/utils.h | ||||
|  * Lorenz Stechauner, 2020-12-03 | ||||
|  * sesimos - secure, simple, modern web server | ||||
|  * @brief Utilities (header file) | ||||
|  * @file src/lib/utils.h | ||||
|  * @author Lorenz Stechauner | ||||
|  * @date 2020-12-03 | ||||
|  */ | ||||
|  | ||||
| #ifndef NECRONDA_SERVER_UTILS_H | ||||
| #define NECRONDA_SERVER_UTILS_H | ||||
| #ifndef SESIMOS_UTILS_H | ||||
| #define SESIMOS_UTILS_H | ||||
|  | ||||
| #include <stdio.h> | ||||
|  | ||||
| @@ -20,9 +21,6 @@ | ||||
|  | ||||
| extern char *log_prefix; | ||||
|  | ||||
| static const char base64_encode_table[64] = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/"; | ||||
| static const int base64_mod_table[3] = {0, 2, 1}; | ||||
|  | ||||
|  | ||||
| #define out_1(fmt) fprintf(stdout, "%s" fmt "\n", log_prefix) | ||||
| #define out_2(fmt, args...) fprintf(stdout, "%s" fmt "\n", log_prefix, args) | ||||
| @@ -36,9 +34,9 @@ static const int base64_mod_table[3] = {0, 2, 1}; | ||||
|  | ||||
| char *format_duration(unsigned long micros, char *buf); | ||||
|  | ||||
| int url_encode_component(const char *str, char *enc, long *size); | ||||
| int url_encode_component(const void *in, size_t size_in, char *out, size_t size_out); | ||||
|  | ||||
| int url_encode(const char *str, char *enc, long *size); | ||||
| int url_encode(const void *in, size_t size_in, char *out, size_t size_out); | ||||
|  | ||||
| int url_decode(const char *str, char *dec, long *size); | ||||
|  | ||||
| @@ -52,4 +50,4 @@ int str_trim_lws(char **start, char **end); | ||||
|  | ||||
| int base64_encode(void *data, unsigned long data_len, char *output, unsigned long *output_len); | ||||
|  | ||||
| #endif //NECRONDA_SERVER_UTILS_H | ||||
| #endif //SESIMOS_UTILS_H | ||||
|   | ||||
							
								
								
									
										207
									
								
								src/lib/websocket.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										207
									
								
								src/lib/websocket.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,207 @@ | ||||
| /** | ||||
|  * sesimos - secure, simple, modern web server | ||||
|  * @brief WebSocket reverse proxy | ||||
|  * @file src/lib/websocket.c | ||||
|  * @author Lorenz Stechauner | ||||
|  * @date 2022-08-16 | ||||
|  */ | ||||
|  | ||||
| #include "../defs.h" | ||||
| #include "websocket.h" | ||||
| #include "utils.h" | ||||
|  | ||||
| #include <string.h> | ||||
| #include <openssl/sha.h> | ||||
| #include <errno.h> | ||||
| #include <signal.h> | ||||
|  | ||||
| static const char ws_key_uuid[] = "258EAFA5-E914-47DA-95CA-C5AB0DC85B11"; | ||||
| static volatile sig_atomic_t terminate = 0; | ||||
|  | ||||
| void ws_terminate(int _) { | ||||
|     terminate = 1; | ||||
| } | ||||
|  | ||||
| int ws_calc_accept_key(const char *key, char *accept_key) { | ||||
|     if (key == NULL || accept_key == NULL) | ||||
|         return -1; | ||||
|  | ||||
|     char input[256] = ""; | ||||
|     unsigned char output[SHA_DIGEST_LENGTH]; | ||||
|     strcat(input, key); | ||||
|     strcat(input, ws_key_uuid); | ||||
|  | ||||
|     if (SHA1((unsigned char *) input, strlen(input), output) == NULL) { | ||||
|         return -2; | ||||
|     } | ||||
|  | ||||
|     base64_encode(output, sizeof(output), accept_key, NULL); | ||||
|  | ||||
|     return 0; | ||||
| } | ||||
|  | ||||
| int ws_recv_frame_header(sock *s, ws_frame *frame) { | ||||
|     unsigned char buf[12]; | ||||
|  | ||||
|     long ret = sock_recv(s, buf, 2, 0); | ||||
|     if (ret < 0) { | ||||
|         print(ERR_STR "Unable to receive from socket: %s" CLR_STR, strerror(errno)); | ||||
|         return -1; | ||||
|     } else if (ret != 2) { | ||||
|         print(ERR_STR "Unable to receive 2 bytes from socket" CLR_STR); | ||||
|         return -2; | ||||
|     } | ||||
|  | ||||
|     unsigned short bits = (buf[0] << 8) | buf[1]; | ||||
|     frame->f_fin = (bits >> 15) & 1; | ||||
|     frame->f_rsv1 = (bits >> 14) & 1; | ||||
|     frame->f_rsv2 = (bits >> 13) & 1; | ||||
|     frame->f_rsv3 = (bits >> 12) & 1; | ||||
|     frame->opcode = (bits >> 8) & 0xF; | ||||
|     frame->f_mask = (bits >> 7) & 1; | ||||
|     unsigned short len = (bits & 0x7F); | ||||
|  | ||||
|     int remaining = frame->f_mask ? 4 : 0; | ||||
|     if (len == 126) { | ||||
|         remaining += 2; | ||||
|     } else if (len == 127) { | ||||
|         remaining += 8; | ||||
|     } | ||||
|  | ||||
|     ret = sock_recv(s, buf, remaining, 0); | ||||
|     if (ret < 0) { | ||||
|         print(ERR_STR "Unable to receive from socket: %s" CLR_STR, strerror(errno)); | ||||
|         return -1; | ||||
|     } else if (ret != remaining) { | ||||
|         print(ERR_STR "Unable to receive correct number of bytes from socket" CLR_STR); | ||||
|         return -2; | ||||
|     } | ||||
|  | ||||
|     if (len == 126) { | ||||
|         frame->len = (((unsigned long) buf[0]) << 8) | ((unsigned long) buf[1]); | ||||
|     } else if (len == 127) { | ||||
|         frame->len = | ||||
|                 (((unsigned long) buf[0]) << 56) | | ||||
|                 (((unsigned long) buf[1]) << 48) | | ||||
|                 (((unsigned long) buf[2]) << 40) | | ||||
|                 (((unsigned long) buf[3]) << 32) | | ||||
|                 (((unsigned long) buf[4]) << 24) | | ||||
|                 (((unsigned long) buf[5]) << 16) | | ||||
|                 (((unsigned long) buf[6]) << 8) | | ||||
|                 (((unsigned long) buf[7]) << 0); | ||||
|     } else { | ||||
|         frame->len = len; | ||||
|     } | ||||
|  | ||||
|     if (frame->f_mask) memcpy(frame->masking_key, buf + (remaining - 4), 4); | ||||
|  | ||||
|     return 0; | ||||
| } | ||||
|  | ||||
| int ws_send_frame_header(sock *s, ws_frame *frame) { | ||||
|     unsigned char buf[14], *ptr = buf; | ||||
|  | ||||
|     unsigned short len; | ||||
|     if (frame->len > 0x7FFF) { | ||||
|         len = 127; | ||||
|     } else if (frame->len > 125) { | ||||
|         len = 126; | ||||
|     } else { | ||||
|         len = frame->len; | ||||
|     } | ||||
|  | ||||
|     unsigned short bits = | ||||
|             (frame->f_fin << 15) | | ||||
|             (frame->f_rsv1 << 14) | | ||||
|             (frame->f_rsv2 << 13) | | ||||
|             (frame->f_rsv3 << 12) | | ||||
|             (frame->opcode << 8) | | ||||
|             (frame->f_mask << 7) | | ||||
|             len; | ||||
|  | ||||
|     ptr++[0] = bits >> 8; | ||||
|     ptr++[0] = bits & 0xFF; | ||||
|  | ||||
|     if (len >= 126) { | ||||
|         for (int i = (len == 126 ? 2 : 8) - 1; i >= 0; i--) | ||||
|             ptr++[0] = (unsigned char) ((frame->len >> (i * 8)) & 0xFF); | ||||
|     } | ||||
|  | ||||
|     if (frame->f_mask) { | ||||
|         memcpy(ptr, frame->masking_key, 4); | ||||
|         ptr += 4; | ||||
|     } | ||||
|  | ||||
|     long ret = sock_send(s, buf, ptr - buf, frame->len != 0 ? MSG_MORE : 0); | ||||
|     if (ret < 0) { | ||||
|         print(ERR_STR "Unable to send to socket: %s" CLR_STR, strerror(errno)); | ||||
|         return -1; | ||||
|     } else if (ret != ptr - buf) { | ||||
|         print(ERR_STR "Unable to send to socket" CLR_STR); | ||||
|         return -2; | ||||
|     } | ||||
|  | ||||
|     return 0; | ||||
| } | ||||
|  | ||||
| int ws_handle_connection(sock *s1, sock *s2) { | ||||
|     sock *poll_socks[2] = {s1, s2}; | ||||
|     sock *readable[2], *error[2]; | ||||
|     int n_sock = 2, n_readable = 0, n_error = 0; | ||||
|     ws_frame frame; | ||||
|     char buf[CHUNK_SIZE]; | ||||
|     int closes = 0; | ||||
|     long ret; | ||||
|  | ||||
|     signal(SIGINT, ws_terminate); | ||||
|     signal(SIGTERM, ws_terminate); | ||||
|  | ||||
|     while (!terminate && closes != 3) { | ||||
|         ret = sock_poll_read(poll_socks, readable, error, n_sock, &n_readable, &n_error, WS_TIMEOUT * 1000); | ||||
|         if (terminate) { | ||||
|             break; | ||||
|         } else if (ret < 0) { | ||||
|             print(ERR_STR "Unable to poll sockets: %s" CLR_STR, strerror(errno)); | ||||
|             return -1; | ||||
|         } else if (n_readable == 0) { | ||||
|             print(ERR_STR "Connection timed out" CLR_STR); | ||||
|             return -2; | ||||
|         } else if (n_error > 0) { | ||||
|             print(ERR_STR "Peer closed connection" CLR_STR); | ||||
|             return -3; | ||||
|         } | ||||
|  | ||||
|         for (int i = 0; i < n_readable; i++) { | ||||
|             sock *s = readable[i]; | ||||
|             sock *o = (s == s1) ? s2 : s1; | ||||
|             if (ws_recv_frame_header(s, &frame) != 0) return -3; | ||||
|  | ||||
|             // print("WebSocket: Peer %s, Opcode=0x%X, Len=%li", (s == s1) ? "1" : "2", frame.opcode, frame.len); | ||||
|  | ||||
|             if (frame.opcode == 0x8) { | ||||
|                 n_sock--; | ||||
|                 if (s == s1) { | ||||
|                     poll_socks[0] = s2; | ||||
|                     closes |= 1; | ||||
|                 } else { | ||||
|                     closes |= 2; | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             if (ws_send_frame_header(o, &frame) != 0) return -3; | ||||
|  | ||||
|             if (frame.len > 0) { | ||||
|                 ret = sock_splice(o, s, buf, sizeof(buf), frame.len); | ||||
|                 if (ret < 0) { | ||||
|                     print(ERR_STR "Unable to forward data in WebSocket: %s" CLR_STR, strerror(errno)); | ||||
|                     return -4; | ||||
|                 } else if (ret != frame.len) { | ||||
|                     print(ERR_STR "Unable to forward correct number of bytes in WebSocket" CLR_STR); | ||||
|                     return -4; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return 0; | ||||
| } | ||||
							
								
								
									
										35
									
								
								src/lib/websocket.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										35
									
								
								src/lib/websocket.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,35 @@ | ||||
| /** | ||||
|  * sesimos - secure, simple, modern web server | ||||
|  * @brief WebSocket reverse proxy (header file) | ||||
|  * @file src/lib/websocket.h | ||||
|  * @author Lorenz Stechauner | ||||
|  * @date 2022-08-16 | ||||
|  */ | ||||
|  | ||||
| #ifndef SESIMOS_WEBSOCKET_H | ||||
| #define SESIMOS_WEBSOCKET_H | ||||
|  | ||||
| #include "sock.h" | ||||
|  | ||||
| #define WS_TIMEOUT 3600 | ||||
|  | ||||
| typedef struct { | ||||
|     unsigned char f_fin:1; | ||||
|     unsigned char f_rsv1:1; | ||||
|     unsigned char f_rsv2:1; | ||||
|     unsigned char f_rsv3:1; | ||||
|     unsigned char opcode:4; | ||||
|     unsigned char f_mask:1; | ||||
|     unsigned long len; | ||||
|     char masking_key[4]; | ||||
| } ws_frame; | ||||
|  | ||||
| int ws_calc_accept_key(const char *key, char *accept_key); | ||||
|  | ||||
| int ws_recv_frame_header(sock *s, ws_frame *frame); | ||||
|  | ||||
| int ws_send_frame_header(sock *s, ws_frame *frame); | ||||
|  | ||||
| int ws_handle_connection(sock *s1, sock *s2); | ||||
|  | ||||
| #endif //SESIMOS_WEBSOCKET_H | ||||
| @@ -1,23 +0,0 @@ | ||||
| /** | ||||
|  * Necronda Web Server | ||||
|  * Definitions | ||||
|  * src/necronda.h | ||||
|  * Lorenz Stechauner, 2021-05-04 | ||||
|  */ | ||||
|  | ||||
| #ifndef NECRONDA_SERVER_NECRONDA_H | ||||
| #define NECRONDA_SERVER_NECRONDA_H | ||||
|  | ||||
| #define NECRONDA_VERSION "4.5" | ||||
| #define SERVER_STR "Necronda/" NECRONDA_VERSION | ||||
| #define SERVER_STR_HTML "Necronda web server " NECRONDA_VERSION | ||||
|  | ||||
| #ifndef DEFAULT_HOST | ||||
| #   define DEFAULT_HOST "www.necronda.net" | ||||
| #endif | ||||
|  | ||||
| #ifndef SERVER_NAME | ||||
| #   define SERVER_NAME DEFAULT_HOST | ||||
| #endif | ||||
|  | ||||
| #endif //NECRONDA_SERVER_NECRONDA_H | ||||
							
								
								
									
										55
									
								
								src/server.c
									
									
									
									
									
								
							
							
						
						
									
										55
									
								
								src/server.c
									
									
									
									
									
								
							| @@ -1,13 +1,12 @@ | ||||
| /** | ||||
|  * Necronda Web Server | ||||
|  * Main executable | ||||
|  * src/necronda-server.c | ||||
|  * Lorenz Stechauner, 2020-12-03 | ||||
|  * Sesimos - secure, simple, modern web server | ||||
|  * @brief Main executable | ||||
|  * @file src/server.c | ||||
|  * @author Lorenz Stechauner | ||||
|  * @date 2020-12-03 | ||||
|  */ | ||||
|  | ||||
| #define _POSIX_C_SOURCE 199309L | ||||
|  | ||||
| #include "necronda.h" | ||||
| #include "defs.h" | ||||
| #include "server.h" | ||||
| #include "client.h" | ||||
|  | ||||
| @@ -22,7 +21,7 @@ | ||||
| #include <sys/socket.h> | ||||
| #include <signal.h> | ||||
| #include <unistd.h> | ||||
| #include <sys/select.h> | ||||
| #include <poll.h> | ||||
| #include <string.h> | ||||
| #include <errno.h> | ||||
| #include <arpa/inet.h> | ||||
| @@ -34,14 +33,15 @@ | ||||
| #include <openssl/conf.h> | ||||
| #include <dirent.h> | ||||
|  | ||||
| int active = 1; | ||||
|  | ||||
| volatile sig_atomic_t active = 1; | ||||
| const char *config_file; | ||||
| int sockets[NUM_SOCKETS]; | ||||
| pid_t children[MAX_CHILDREN]; | ||||
| MMDB_s mmdbs[MAX_MMDB]; | ||||
| SSL_CTX *contexts[CONFIG_MAX_CERT_CONFIG]; | ||||
|  | ||||
| void openssl_init() { | ||||
| void openssl_init(void) { | ||||
|     SSL_library_init(); | ||||
|     SSL_load_error_strings(); | ||||
|     ERR_load_BIO_strings(); | ||||
| @@ -57,7 +57,7 @@ static int ssl_servername_cb(SSL *ssl, int *ad, void *arg) { | ||||
|     return SSL_TLSEXT_ERR_OK; | ||||
| } | ||||
|  | ||||
| void destroy() { | ||||
| void destroy(int _) { | ||||
|     fprintf(stderr, "\n" ERR_STR "Terminating forcefully!" CLR_STR "\n"); | ||||
|     int status = 0; | ||||
|     int ret; | ||||
| @@ -86,7 +86,7 @@ void destroy() { | ||||
|     exit(2); | ||||
| } | ||||
|  | ||||
| void terminate() { | ||||
| void terminate(int _) { | ||||
|     fprintf(stderr, "\nTerminating gracefully...\n"); | ||||
|     active = 0; | ||||
|  | ||||
| @@ -153,8 +153,7 @@ void terminate() { | ||||
|  | ||||
| int main(int argc, const char *argv[]) { | ||||
|     const int YES = 1; | ||||
|     fd_set socket_fds, read_socket_fds; | ||||
|     int max_socket_fd = 0; | ||||
|     struct pollfd poll_fds[NUM_SOCKETS]; | ||||
|     int ready_sockets_num; | ||||
|     long client_num = 0; | ||||
|     char buf[1024]; | ||||
| @@ -169,8 +168,6 @@ int main(int argc, const char *argv[]) { | ||||
|     memset(children, 0, sizeof(children)); | ||||
|     memset(mmdbs, 0, sizeof(mmdbs)); | ||||
|  | ||||
|     struct timeval timeout; | ||||
|  | ||||
|     const struct sockaddr_in6 addresses[2] = { | ||||
|             {.sin6_family = AF_INET6, .sin6_addr = IN6ADDR_ANY_INIT, .sin6_port = htons(80)}, | ||||
|             {.sin6_family = AF_INET6, .sin6_addr = IN6ADDR_ANY_INIT, .sin6_port = htons(443)} | ||||
| @@ -180,7 +177,7 @@ int main(int argc, const char *argv[]) { | ||||
|         fprintf(stderr, ERR_STR "Unable to set stdout to line-buffered mode: %s" CLR_STR, strerror(errno)); | ||||
|         return 1; | ||||
|     } | ||||
|     printf("Necronda Web Server " NECRONDA_VERSION "\n"); | ||||
|     printf("Sesimos web server " SERVER_VERSION "\n"); | ||||
|  | ||||
|     ret = config_init(); | ||||
|     if (ret != 0) { | ||||
| @@ -191,7 +188,7 @@ int main(int argc, const char *argv[]) { | ||||
|     for (int i = 1; i < argc; i++) { | ||||
|         const char *arg = argv[i]; | ||||
|         if (strcmp(arg, "-h") == 0 || strcmp(arg, "--help") == 0) { | ||||
|             printf("Usage: necronda-server [-h] [-c <CONFIG-FILE>]\n" | ||||
|             printf("Usage: sesimos [-h] [-c <CONFIG-FILE>]\n" | ||||
|                    "\n" | ||||
|                    "Options:\n" | ||||
|                    "  -c, --config <CONFIG-FILE>  path to the config file. If not provided, default will be used\n" | ||||
| @@ -292,10 +289,6 @@ int main(int argc, const char *argv[]) { | ||||
|  | ||||
|     openssl_init(); | ||||
|  | ||||
|     client.buf = NULL; | ||||
|     client.buf_len = 0; | ||||
|     client.buf_off = 0; | ||||
|  | ||||
|     for (int i = 0; i < CONFIG_MAX_CERT_CONFIG; i++) { | ||||
|         const cert_config *conf = &config->certs[i]; | ||||
|         if (conf->name[0] == 0) break; | ||||
| @@ -338,29 +331,23 @@ int main(int argc, const char *argv[]) { | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     FD_ZERO(&socket_fds); | ||||
|     for (int i = 0; i < NUM_SOCKETS; i++) { | ||||
|         FD_SET(sockets[i], &socket_fds); | ||||
|         if (sockets[i] > max_socket_fd) { | ||||
|             max_socket_fd = sockets[i]; | ||||
|         } | ||||
|         poll_fds[i].fd = sockets[i]; | ||||
|         poll_fds[i].events = POLLIN; | ||||
|     } | ||||
|  | ||||
|     fprintf(stderr, "Ready to accept connections\n"); | ||||
|  | ||||
|     while (active) { | ||||
|         timeout.tv_sec = 1; | ||||
|         timeout.tv_usec = 0; | ||||
|         read_socket_fds = socket_fds; | ||||
|         ready_sockets_num = select(max_socket_fd + 1, &read_socket_fds, NULL, NULL, &timeout); | ||||
|         ready_sockets_num = poll(poll_fds, NUM_SOCKETS, 1000); | ||||
|         if (ready_sockets_num < 0) { | ||||
|             fprintf(stderr, ERR_STR "Unable to select sockets: %s" CLR_STR "\n", strerror(errno)); | ||||
|             terminate(); | ||||
|             fprintf(stderr, ERR_STR "Unable to poll sockets: %s" CLR_STR "\n", strerror(errno)); | ||||
|             terminate(0); | ||||
|             return 1; | ||||
|         } | ||||
|  | ||||
|         for (int i = 0; i < NUM_SOCKETS; i++) { | ||||
|             if (FD_ISSET(sockets[i], &read_socket_fds)) { | ||||
|             if (poll_fds[i].revents & POLLIN) { | ||||
|                 client_fd = accept(sockets[i], (struct sockaddr *) &client_addr, &client_addr_len); | ||||
|                 if (client_fd < 0) { | ||||
|                     fprintf(stderr, ERR_STR "Unable to accept connection: %s" CLR_STR "\n", strerror(errno)); | ||||
|   | ||||
							
								
								
									
										19
									
								
								src/server.h
									
									
									
									
									
								
							
							
						
						
									
										19
									
								
								src/server.h
									
									
									
									
									
								
							| @@ -1,15 +1,17 @@ | ||||
| /** | ||||
|  * Necronda Web Server | ||||
|  * Main executable (header file) | ||||
|  * src/necronda-server.h | ||||
|  * Lorenz Stechauner, 2020-12-03 | ||||
|  * sesimos - secure, simple, modern web server | ||||
|  * @brief Main executable (header file) | ||||
|  * @file src/server.h | ||||
|  * @author Lorenz Stechauner | ||||
|  * @date 2020-12-03 | ||||
|  */ | ||||
|  | ||||
| #ifndef NECRONDA_SERVER_SERVER_H | ||||
| #define NECRONDA_SERVER_SERVER_H | ||||
| #ifndef SESIMOS_SERVER_H | ||||
| #define SESIMOS_SERVER_H | ||||
|  | ||||
| #include <sys/time.h> | ||||
| #include <maxminddb.h> | ||||
| #include <signal.h> | ||||
|  | ||||
| #define NUM_SOCKETS 2 | ||||
| #define MAX_CHILDREN 1024 | ||||
| @@ -20,15 +22,14 @@ | ||||
| #define SERVER_TIMEOUT_INIT 4 | ||||
| #define SERVER_TIMEOUT 3600 | ||||
|  | ||||
| #define CHUNK_SIZE 8192 | ||||
|  | ||||
| extern int sockets[NUM_SOCKETS]; | ||||
| extern pid_t children[MAX_CHILDREN]; | ||||
| extern MMDB_s mmdbs[MAX_MMDB]; | ||||
|  | ||||
| extern int server_keep_alive; | ||||
| extern volatile sig_atomic_t 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_SERVER_H | ||||
| #endif //SESIMOS_SERVER_H | ||||
|   | ||||
							
								
								
									
										28
									
								
								test/mock_socket.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								test/mock_socket.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,28 @@ | ||||
|  | ||||
| #include <stdio.h> | ||||
| #include <errno.h> | ||||
|  | ||||
| #include "mock_socket.h" | ||||
|  | ||||
| int mock_socket_send_mode; | ||||
|  | ||||
| static int sockets[256] = {0}; | ||||
| static int n_sockets = 0; | ||||
|  | ||||
| int mock_socket(int domain, int type, int protocol) { | ||||
|     printf("SOCKET\n"); | ||||
|     return (n_sockets++) + 100; | ||||
| } | ||||
|  | ||||
| ssize_t mock_send(int fd, const void *buf, size_t n, int flags) { | ||||
|     printf("SEND\n"); | ||||
|     if (mock_socket_send_mode == MOCK_SOCKET_MODE_EINTR) { | ||||
|         errno = EINTR; | ||||
|         return rand() % ((ssize_t) n) - 1; | ||||
|     } else if (mock_socket_send_mode == MOCK_SOCKET_MODE_CLOSED) { | ||||
|         errno = 0; // TODO | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     return (ssize_t) n; | ||||
| } | ||||
							
								
								
									
										20
									
								
								test/mock_socket.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								test/mock_socket.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,20 @@ | ||||
|  | ||||
| #ifndef SESIMOS_MOCK_SOCKET_H | ||||
| #define SESIMOS_MOCK_SOCKET_H | ||||
|  | ||||
| #include <stdlib.h> | ||||
|  | ||||
| #define MOCK_SOCKET_MODE_SUCCESS 0 | ||||
| #define MOCK_SOCKET_MODE_EINTR 1 | ||||
| #define MOCK_SOCKET_MODE_CLOSED 2 | ||||
|  | ||||
| #define socket(args...) mock_socket(args) | ||||
| #define send(args...) mock_send(args) | ||||
|  | ||||
| extern int mock_socket_send_mode; | ||||
|  | ||||
| int mock_socket(int domain, int type, int protocol); | ||||
|  | ||||
| ssize_t mock_send(int fd, const void *buf, size_t n, int flags); | ||||
|  | ||||
| #endif //SESIMOS_MOCK_SOCKET_H | ||||
							
								
								
									
										33
									
								
								test/mock_ssl.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								test/mock_ssl.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,33 @@ | ||||
|  | ||||
| #include <openssl/crypto.h> | ||||
|  | ||||
|  | ||||
| int SSL_write(SSL *ssl, const void *buf, int num) { | ||||
|     return num; | ||||
| } | ||||
|  | ||||
| int SSL_read(SSL *ssl, void *buf, int num) { | ||||
|     return num; | ||||
| } | ||||
|  | ||||
| int SSL_peek(SSL *ssl, void *buf, int num) { | ||||
|     return num; | ||||
| } | ||||
|  | ||||
| int SSL_get_error(const SSL *s, int ret_code) { | ||||
|     return 0; | ||||
| } | ||||
|  | ||||
| const char *ERR_reason_error_string(unsigned long e) { | ||||
|     return ""; | ||||
| } | ||||
|  | ||||
| int SSL_shutdown(SSL *s) { | ||||
|     return 0; | ||||
| } | ||||
|  | ||||
| void SSL_free(SSL *ssl) {} | ||||
|  | ||||
| unsigned long ERR_get_error(void) { | ||||
|     return 0; | ||||
| } | ||||
							
								
								
									
										15
									
								
								test/test_sock.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								test/test_sock.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,15 @@ | ||||
|  | ||||
| #include <criterion/criterion.h> | ||||
| #include "mock_socket.h" | ||||
| #include "../src/lib/sock.h" | ||||
|  | ||||
|  | ||||
| Test(sock, sock_send_1) { | ||||
|     int fd = socket(AF_INET6, SOCK_STREAM, 0); | ||||
|     sock s; | ||||
|     s.enc = 0; | ||||
|     s.socket = fd; | ||||
|  | ||||
|     long ret = sock_send(&s, "Hello", 5, 0); | ||||
|     cr_assert_eq(ret, 5); | ||||
| } | ||||
							
								
								
									
										86
									
								
								test/test_utils.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										86
									
								
								test/test_utils.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,86 @@ | ||||
|  | ||||
| #include <criterion/criterion.h> | ||||
| #include <criterion/parameterized.h> | ||||
|  | ||||
| #include "../src/lib/utils.h" | ||||
|  | ||||
| struct url_encode_t { | ||||
|     long in_size; | ||||
|     char in[256]; | ||||
|     long exp_size; | ||||
|     char exp[256]; | ||||
| }; | ||||
|  | ||||
| struct format_duration_t { | ||||
|     unsigned long micros; | ||||
|     char exp[16]; | ||||
| }; | ||||
|  | ||||
| ParameterizedTestParameters(utils, url_encode) { | ||||
|     static struct url_encode_t params[] = { | ||||
|             {0, "", 0, ""}, | ||||
|             {9, "Test Text", 11, "Test%20Text"}, | ||||
|             {21, "Text\0with\0null\0bytes\0", 29, "Text%00with%00null%00bytes%00"}, | ||||
|             {59, "Text&with+some/strange_symbols-or#something?I%don't|know...", 59, "Text&with+some/strange_symbols-or#something?I%don't|know..."}, | ||||
|             {33, "Data\x12With\x13Some" "\xFF" "Control" "\xFE" "Characters", 41, "Data%12With%13Some%FFControl%FECharacters"} | ||||
|     }; | ||||
|     return cr_make_param_array(struct url_encode_t, params, sizeof(params) / sizeof(struct url_encode_t)); | ||||
| } | ||||
|  | ||||
| ParameterizedTest(struct url_encode_t *param, utils, url_encode) { | ||||
|     char out[256]; | ||||
|     cr_assert_eq(url_encode(param->in, param->in_size, out, sizeof(out)), param->exp_size); | ||||
|     cr_assert_arr_eq(out, param->exp, param->exp_size + 1); | ||||
| } | ||||
|  | ||||
| Test(utils, url_encode_bytes) { | ||||
|     char out[4]; | ||||
|     char exp[4]; | ||||
|  | ||||
|     for (int i = 0; i < 256; i++) { | ||||
|         unsigned char ch = i; | ||||
|         if (ch <= 0x20 || ch >= 0x7F) { | ||||
|             cr_assert_eq(url_encode(&ch, 1, out, sizeof(out)), 3); | ||||
|             sprintf(exp, "%%%02X", ch); | ||||
|             cr_assert_str_eq(out, exp); | ||||
|         } else { | ||||
|             cr_assert_eq(url_encode(&ch, 1, out, sizeof(out)), 1); | ||||
|             sprintf(exp, "%c", ch); | ||||
|             cr_assert_str_eq(out, exp); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| Test(utils, url_encode_invalid) { | ||||
|     cr_assert_eq(url_encode("Hello", 5, NULL, 0), 5); | ||||
| } | ||||
|  | ||||
| ParameterizedTestParameters(utils, format_duration) { | ||||
|     static struct format_duration_t params[] = { | ||||
|             {0, "0.0 ms"}, | ||||
|             {1, "0.0 ms"}, | ||||
|             {90, "0.1 ms"}, | ||||
|             {100, "0.1 ms"}, | ||||
|             {110, "0.1 ms"}, | ||||
|             {900, "0.9 ms"}, | ||||
|             {1000, "1.0 ms"}, | ||||
|             {9000, "9.0 ms"}, | ||||
|             {9899, "9.9 ms"}, | ||||
|             {9999, "10.0 ms"}, | ||||
|             {10000, "10 ms"}, | ||||
|             {11999, "12 ms"}, | ||||
|             {999999, "1.0 s"}, | ||||
|             {1000000, "1.0 s"}, | ||||
|             {3000000, "3.0 s"}, | ||||
|             {1000000 * 60, "1.0 min"}, | ||||
|             {1000000 * 60 * 30L - 30000000, "29.5 min"}, | ||||
|             {1000000 * 60 * 60L, "60.0 min"}, | ||||
|             {1000000 * 60 * 120L, "120 min"}, | ||||
|     }; | ||||
|     return cr_make_param_array(struct format_duration_t, params, sizeof(params) / sizeof(struct format_duration_t)); | ||||
| } | ||||
|  | ||||
| ParameterizedTest(struct format_duration_t *param, utils, format_duration) { | ||||
|     char buf[16]; | ||||
|     cr_assert_str_eq(format_duration(param->micros, buf), param->exp); | ||||
| } | ||||
		Reference in New Issue
	
	Block a user