Home | History | Annotate | Line # | Download | only in httpd
      1 /*	$NetBSD: bozohttpd.h,v 1.73 2022/09/12 10:30:39 martin Exp $	*/
      2 
      3 /*	$eterna: bozohttpd.h,v 1.39 2011/11/18 09:21:15 mrg Exp $	*/
      4 
      5 /*
      6  * Copyright (c) 1997-2022 Matthew R. Green
      7  * All rights reserved.
      8  *
      9  * Redistribution and use in source and binary forms, with or without
     10  * modification, are permitted provided that the following conditions
     11  * are met:
     12  * 1. Redistributions of source code must retain the above copyright
     13  *    notice, this list of conditions and the following disclaimer.
     14  * 2. Redistributions in binary form must reproduce the above copyright
     15  *    notice, this list of conditions and the following disclaimer and
     16  *    dedication in the documentation and/or other materials provided
     17  *    with the distribution.
     18  *
     19  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
     20  * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
     21  * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
     22  * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
     23  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
     24  * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
     25  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
     26  * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
     27  * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
     28  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
     29  * SUCH DAMAGE.
     30  *
     31  */
     32 #ifndef BOZOHTTPD_H_
     33 #define BOZOHTTPD_H_	1
     34 
     35 #include "netbsd_queue.h"
     36 
     37 #include <stdbool.h>
     38 #include <stdio.h>
     39 #include <signal.h>
     40 
     41 #include <sys/stat.h>
     42 
     43 #ifndef NO_LUA_SUPPORT
     44 #include <lua.h>
     45 #endif
     46 
     47 #ifndef NO_BLOCKLIST_SUPPORT
     48 #include <blocklist.h>
     49 void pfilter_notify(const int, const int);
     50 #endif
     51 
     52 /* QNX provides a lot of NetBSD things in nbutil.h */
     53 #ifdef HAVE_NBUTIL_H
     54 #include <nbutil.h>
     55 #endif
     56 
     57 /* lots of "const" but gets free()'ed etc at times, sigh */
     58 
     59 /* headers */
     60 typedef struct bozoheaders {
     61 	/*const*/ char *h_header;
     62 	/*const*/ char *h_value;	/* this gets free()'ed etc at times */
     63 	SIMPLEQ_ENTRY(bozoheaders)	h_next;
     64 } bozoheaders_t;
     65 SIMPLEQ_HEAD(qheaders, bozoheaders);
     66 
     67 #ifndef NO_LUA_SUPPORT
     68 typedef struct lua_handler {
     69 	const char	*name;
     70 	int		 ref;
     71 	SIMPLEQ_ENTRY(lua_handler)	h_next;
     72 } lua_handler_t;
     73 
     74 typedef struct lua_state_map {
     75 	const char 	*script;
     76 	const char	*prefix;
     77 	lua_State	*L;
     78 	SIMPLEQ_HEAD(, lua_handler)	handlers;
     79 	SIMPLEQ_ENTRY(lua_state_map)	s_next;
     80 } lua_state_map_t;
     81 #endif
     82 
     83 typedef struct bozo_content_map_t {
     84 	const char	*name;		/* postfix of file */
     85 	const char	*type;		/* matching content-type */
     86 	const char	*encoding;	/* matching content-encoding */
     87 	const char	*encoding11;	/* matching content-encoding (HTTP/1.1) */
     88 	const char	*cgihandler;	/* optional CGI handler */
     89 } bozo_content_map_t;
     90 
     91 /* this struct holds the bozo constants */
     92 typedef struct bozo_consts_t {
     93 	const char	*http_09;	/* "HTTP/0.9" */
     94 	const char	*http_10;	/* "HTTP/1.0" */
     95 	const char	*http_11;	/* "HTTP/1.1" */
     96 	const char	*text_plain;	/* "text/plain" */
     97 } bozo_consts_t;
     98 
     99 /* this structure encapsulates all the bozo flags and control vars */
    100 typedef struct bozohttpd_t {
    101 	char		*rootdir;	/* root directory */
    102 	int		 numeric;	/* avoid gethostby*() */
    103 	char		*virtbase;	/* virtual directory base */
    104 	int		 unknown_slash;	/* unknown vhosts go to normal slashdir */
    105 	int		 logstderr;	/* log to stderr (even if not tty) */
    106 	int		 nolog;		/* do not log anything */
    107 	int		 background;	/* drop into daemon mode */
    108 	int		 foreground;	/* keep daemon mode in foreground */
    109 	char		*pidfile;	/* path to the pid file, if any */
    110 	size_t		 page_size;	/* page size */
    111 	char		*slashdir;	/* www slash directory */
    112 	char		*bindport;	/* bind port; default "http" */
    113 	char		*bindaddress;	/* address for binding - INADDR_ANY */
    114 	int		 debug;		/* debugging level */
    115 	char		*virthostname;	/* my name */
    116 	const char	*server_software;/* our brand :-) */
    117 	const char	*index_html;	/* our home page */
    118 	const char	*public_html;	/* ~user/public_html page */
    119 	int		 enable_users;	/* enable public_html */
    120 	int		 enable_cgi_users;	/* use the cgi handler */
    121 	int		*sock;		/* bound sockets */
    122 	int		 nsock;		/* number of above */
    123 	struct pollfd	*fds;		/* current poll fd set */
    124 	int		 request_times;	/* # times a request was processed */
    125 	int		 dir_indexing;	/* handle directories */
    126 	const char	*dir_readme;	/* include README footer in indexes */
    127 	int		 hide_dots;	/* hide .* */
    128 	int		 process_cgi;	/* use the cgi handler */
    129 	char		*cgibin;	/* cgi-bin directory */
    130 	unsigned	ssl_timeout;	/* ssl timeout */
    131 	unsigned	initial_timeout;/* first line timeout */
    132 	unsigned	header_timeout;	/* header lines timeout */
    133 	unsigned	request_timeout;/* total session timeout */
    134 	char		*ssl_min_proto;	/* minimum ssl protocol level */
    135 #ifndef NO_LUA_SUPPORT
    136 	int		 process_lua;	/* use the Lua handler */
    137 	SIMPLEQ_HEAD(, lua_state_map)	lua_states;
    138 #endif
    139 	void		*sslinfo;	/* pointer to ssl struct */
    140 	int		dynamic_content_map_size;/* size of dyn cont map */
    141 	bozo_content_map_t	*dynamic_content_map;/* dynamic content map */
    142 	size_t		 mmapsz;	/* size of region to mmap */
    143 	char		*getln_buffer;	/* space for getln buffer */
    144 	ssize_t		 getln_buflen;	/* length of allocated space */
    145 	char		*errorbuf;	/* no dynamic allocation allowed */
    146 	bozo_consts_t	 consts;	/* various constants */
    147 } bozohttpd_t;
    148 
    149 /* bozo_httpreq_t */
    150 typedef struct bozo_httpreq_t {
    151 	bozohttpd_t	*hr_httpd;
    152 	int		hr_method;
    153 #define	HTTP_GET	0x01
    154 #define HTTP_POST	0x02
    155 #define HTTP_HEAD	0x03
    156 #define HTTP_OPTIONS	0x04	/* not supported */
    157 #define HTTP_PUT	0x05	/* not supported */
    158 #define HTTP_DELETE	0x06	/* not supported */
    159 #define HTTP_TRACE	0x07	/* not supported */
    160 #define HTTP_CONNECT	0x08	/* not supported */
    161 	const char *hr_methodstr;
    162 	char	*hr_virthostname;	/* server name (if not identical
    163 					   to hr_httpd->virthostname) */
    164 	char	*hr_file_free;	/* pointer to file buffer to free() */
    165 	char	*hr_file;	/* pointer into file buffer */
    166 	char	*hr_oldfile;	/* if we added an index_html */
    167 	char	*hr_query;
    168 	char	*hr_host;	/* HTTP/1.1 Host: or virtual hostname,
    169 				   possibly including a port number */
    170 #ifndef NO_USER_SUPPORT
    171 	char	*hr_user;	/* username if we hit userdir request */
    172 #endif /* !NO_USER_SUPPORT */
    173 	const char *hr_proto;
    174 	const char *hr_content_type;
    175 	const char *hr_content_length;
    176 	const char *hr_allow;
    177 	const char *hr_referrer;
    178 	const char *hr_range;
    179 	const char *hr_if_modified_since;
    180 	const char *hr_accept_encoding;
    181 	int         hr_have_range;
    182 	off_t       hr_first_byte_pos;
    183 	off_t       hr_last_byte_pos;
    184 	/*const*/ char *hr_remotehost;
    185 	/*const*/ char *hr_remoteaddr;
    186 	/*const*/ char *hr_serverport;
    187 #ifdef DO_HTPASSWD
    188 	/*const*/ char *hr_authrealm;
    189 	/*const*/ char *hr_authuser;
    190 	/*const*/ char *hr_authpass;
    191 #endif
    192 	struct qheaders		hr_headers;
    193 	struct qheaders		hr_replheaders;
    194 	unsigned		hr_nheaders;
    195 	size_t			hr_header_bytes;
    196 } bozo_httpreq_t;
    197 
    198 /* helper to access the "active" host name from a httpd/request pair */
    199 #define	BOZOHOST(HTTPD,REQUEST)	((REQUEST)->hr_virthostname ?		\
    200 					(REQUEST)->hr_virthostname :	\
    201 					(HTTPD)->virthostname)
    202 
    203 /* structure to hold string based (name, value) pairs with preferences */
    204 typedef struct bozoprefs_t {
    205 	size_t		  size;		/* size of the two arrays */
    206 	size_t		  count;	/* # of entries in arrays */
    207 	char		**name;		/* names of each entry */
    208 	char		**value;	/* values for the name entries */
    209 } bozoprefs_t;
    210 
    211 /* sun2 has a tiny VA range */
    212 #ifdef __mc68010__
    213 #ifndef BOZO_WRSZ
    214 #define BOZO_WRSZ	(16 * 1024)
    215 #endif
    216 #ifndef BOZO_MMAPSZ
    217 #define BOZO_MMAPSZ	(BOZO_WRSZ * 4)
    218 #endif
    219 #endif
    220 
    221 /* by default write in upto 64KiB chunks, and mmap in upto 64MiB chunks */
    222 #ifndef BOZO_WRSZ
    223 #define BOZO_WRSZ	(64 * 1024)
    224 #endif
    225 #ifndef BOZO_MMAPSZ
    226 #define BOZO_MMAPSZ	(BOZO_WRSZ * 1024)
    227 #endif
    228 
    229 /* only allow this many total headers bytes */
    230 #define BOZO_HEADERS_MAX_SIZE (16 * 1024)
    231 
    232 #define BOZO_MINBUFSIZE (4 * 1024)
    233 
    234 /* debug flags */
    235 #define DEBUG_NORMAL	1
    236 #define DEBUG_FAT	2
    237 #define DEBUG_OBESE	3
    238 #define DEBUG_EXPLODING	4
    239 
    240 #define	strornull(x)	((x) ? (x) : "<null>")
    241 
    242 #if (defined(__GNUC__) && __GNUC__ >= 3) || defined(__lint__)
    243 #define BOZO_PRINTFLIKE(x,y) __attribute__((__format__(__printf__, x,y)))
    244 #define BOZO_DEAD __attribute__((__noreturn__))
    245 #define BOZO_CHECKRET __attribute__((__warn_unused_result__))
    246 #else
    247 #define BOZO_PRINTFLIKE(x,y)
    248 #define BOZO_DEAD
    249 #define BOZO_CHECKRET
    250 #endif
    251 
    252 #ifdef NO_DEBUG
    253 #define	debug(x)
    254 #define have_debug	(0)
    255 #else
    256 void	debug__(bozohttpd_t *, int, const char *, ...) BOZO_PRINTFLIKE(3, 4);
    257 #define debug(x)	debug__ x
    258 #define have_debug	(1)
    259 #endif /* NO_DEBUG */
    260 
    261 #define BOZO_HTTP_PORT	"80"
    262 #define BOZO_HTTPS_PORT	"443"
    263 
    264 /*
    265  * bozohttpd special files.  avoid serving these out.
    266  *
    267  * When you add some .bz* file, make sure to also check it in
    268  * bozo_check_special_files()
    269  */
    270 
    271 #ifndef REDIRECT_FILE
    272 #define REDIRECT_FILE		".bzredirect"
    273 #endif
    274 #ifndef ABSREDIRECT_FILE
    275 #define ABSREDIRECT_FILE	".bzabsredirect"
    276 #endif
    277 #ifndef REMAP_FILE
    278 #define REMAP_FILE		".bzremap"
    279 #endif
    280 #ifndef AUTH_FILE
    281 #define AUTH_FILE		".htpasswd"
    282 #endif
    283 
    284 /* be sure to always return this error up */
    285 int	bozo_http_error(bozohttpd_t *, int, bozo_httpreq_t *, const char *);
    286 
    287 int	bozo_check_special_files(bozo_httpreq_t *, const char *, bool) BOZO_CHECKRET;
    288 char	*bozo_http_date(char *, size_t);
    289 void	bozo_print_header(bozo_httpreq_t *, struct stat *, const char *,
    290 			  const char *);
    291 char	*bozo_escape_rfc3986(bozohttpd_t *httpd, const char *url, int absolute);
    292 char	*bozo_escape_html(bozohttpd_t *httpd, const char *url);
    293 int	bozo_decode_url_percent(bozo_httpreq_t *, char *);
    294 
    295 /* these are similar to libc functions, no underscore here */
    296 void	bozowarn(bozohttpd_t *, const char *, ...)
    297 		BOZO_PRINTFLIKE(2, 3);
    298 void	bozoerr(bozohttpd_t *, int, const char *, ...)
    299 		BOZO_PRINTFLIKE(3, 4)
    300 		BOZO_DEAD;
    301 void	bozoasprintf(bozohttpd_t *, char **, const char *, ...)
    302 		BOZO_PRINTFLIKE(3, 4);
    303 char	*bozodgetln(bozohttpd_t *, int, ssize_t *, ssize_t (*)(bozohttpd_t *,
    304 		    int, void *, size_t));
    305 char	*bozostrnsep(char **, const char *, ssize_t *);
    306 void	*bozomalloc(bozohttpd_t *, size_t);
    307 void	*bozorealloc(bozohttpd_t *, void *, size_t);
    308 char	*bozostrdup(bozohttpd_t *, bozo_httpreq_t *, const char *);
    309 
    310 #define bozo_noop	do { /* nothing */ } while (/*CONSTCOND*/0)
    311 
    312 #define have_core					(1)
    313 
    314 /* ssl-bozo.c */
    315 #ifdef NO_SSL_SUPPORT
    316 #define bozo_ssl_set_opts(w, x, y)			bozo_noop
    317 #define bozo_ssl_set_ciphers(w, x)			bozo_noop
    318 #define bozo_ssl_init(x)				bozo_noop
    319 #define bozo_ssl_accept(x)				(0)
    320 #define bozo_ssl_shutdown(x)				bozo_noop
    321 #define bozo_ssl_destroy(x)				bozo_noop
    322 #define have_ssl					(0)
    323 #else
    324 void	bozo_ssl_set_opts(bozohttpd_t *, const char *, const char *);
    325 void	bozo_ssl_set_ciphers(bozohttpd_t *, const char *);
    326 void	bozo_ssl_init(bozohttpd_t *);
    327 int	bozo_ssl_accept(bozohttpd_t *);
    328 void	bozo_ssl_shutdown(bozohttpd_t *);
    329 void	bozo_ssl_destroy(bozohttpd_t *);
    330 #define have_ssl					(1)
    331 #endif
    332 
    333 
    334 /* auth-bozo.c */
    335 #ifdef DO_HTPASSWD
    336 void	bozo_auth_init(bozo_httpreq_t *);
    337 int	bozo_auth_check(bozo_httpreq_t *, const char *);
    338 void	bozo_auth_cleanup(bozo_httpreq_t *);
    339 int	bozo_auth_check_headers(bozo_httpreq_t *, char *, char *, ssize_t);
    340 void	bozo_auth_check_401(bozo_httpreq_t *, int);
    341 void	bozo_auth_cgi_setenv(bozo_httpreq_t *, char ***);
    342 int	bozo_auth_cgi_count(bozo_httpreq_t *);
    343 #else
    344 #define	bozo_auth_init(x)				bozo_noop
    345 #define	bozo_auth_check(x, y)				(0)
    346 #define	bozo_auth_cleanup(x)				bozo_noop
    347 #define	bozo_auth_check_headers(y, z, a, b)		(0)
    348 #define	bozo_auth_check_401(x, y)			bozo_noop
    349 #define	bozo_auth_cgi_setenv(x, y)			bozo_noop
    350 #define	bozo_auth_cgi_count(x)				(0)
    351 #endif /* DO_HTPASSWD */
    352 
    353 
    354 /* cgi-bozo.c */
    355 #ifdef NO_CGIBIN_SUPPORT
    356 #define bozo_cgi_setbin(h,s)				bozo_noop
    357 #define	bozo_process_cgi(h)				(0)
    358 #define have_cgibin					(0)
    359 #else
    360 void	bozo_cgi_setbin(bozohttpd_t *, const char *);
    361 void	bozo_setenv(bozohttpd_t *, const char *, const char *, char **);
    362 int	bozo_process_cgi(bozo_httpreq_t *);
    363 #define have_cgibin					(1)
    364 #endif /* NO_CGIBIN_SUPPORT */
    365 
    366 
    367 /* lua-bozo.c */
    368 #ifdef NO_LUA_SUPPORT
    369 #define bozo_process_lua(h)				(0)
    370 #define bozo_add_lua_map(h,s,t)				bozo_noop
    371 #define have_lua					(0)
    372 #else
    373 void	bozo_add_lua_map(bozohttpd_t *, const char *, const char *);
    374 int	bozo_process_lua(bozo_httpreq_t *);
    375 #define have_lua					(1)
    376 #endif /* NO_LUA_SUPPORT */
    377 
    378 
    379 /* daemon-bozo.c */
    380 #ifdef NO_DAEMON_MODE
    381 #define bozo_daemon_init(x)				bozo_noop
    382 #define bozo_daemon_fork(x)				(0)
    383 #define bozo_daemon_closefds(x)				bozo_noop
    384 #define have_daemon_mode				(0)
    385 #else
    386 void	bozo_daemon_init(bozohttpd_t *);
    387 int	bozo_daemon_fork(bozohttpd_t *);
    388 void	bozo_daemon_closefds(bozohttpd_t *);
    389 #define have_daemon_mode				(1)
    390 #endif /* NO_DAEMON_MODE */
    391 
    392 
    393 /* tilde-luzah-bozo.c */
    394 #ifdef NO_USER_SUPPORT
    395 #define bozo_user_transform(x)				(0)
    396 #define bozo_user_free(x)				/* nothing */
    397 #define have_user					(0)
    398 #else
    399 int	bozo_user_transform(bozo_httpreq_t *);
    400 #define bozo_user_free(x)				free(x)
    401 #define have_user					(1)
    402 #endif /* NO_USER_SUPPORT */
    403 
    404 
    405 /* dir-index-bozo.c */
    406 #ifdef NO_DIRINDEX_SUPPORT
    407 #define bozo_dir_index(a, b, c)				(0)
    408 #define have_dirindex					(0)
    409 #else
    410 int	bozo_dir_index(bozo_httpreq_t *, const char *, int);
    411 #define have_dirindex					(1)
    412 #endif /* NO_DIRINDEX_SUPPORT */
    413 
    414 
    415 /* content-bozo.c */
    416 const char *bozo_content_type(bozo_httpreq_t *, const char *);
    417 const char *bozo_content_encoding(bozo_httpreq_t *, const char *);
    418 bozo_content_map_t *bozo_match_content_map(bozohttpd_t *, const char *, int);
    419 bozo_content_map_t *bozo_get_content_map(bozohttpd_t *, const char *);
    420 #ifdef NO_DYNAMIC_CONTENT
    421 #define bozo_add_content_map_mime(h,s,t,u,v)		bozo_noop
    422 #define have_dynamic_content				(0)
    423 #else
    424 void	bozo_add_content_map_mime(bozohttpd_t *, const char *, const char *,
    425 				  const char *, const char *);
    426 #define have_dynamic_content				(1)
    427 #endif
    428 
    429 /* additional cgi-bozo.c */
    430 #if have_cgibin && have_dynamic_content
    431 void	bozo_add_content_map_cgi(bozohttpd_t *, const char *, const char *);
    432 #else
    433 #define	bozo_add_content_map_cgi(h,s,t)
    434 #endif
    435 
    436 /* I/O */
    437 int bozo_printf(bozohttpd_t *, const char *, ...) BOZO_PRINTFLIKE(2, 3);
    438 ssize_t bozo_read(bozohttpd_t *, int, void *, size_t);
    439 ssize_t bozo_write(bozohttpd_t *, int, const void *, size_t);
    440 int bozo_flush(bozohttpd_t *, FILE *);
    441 
    442 /* misc */
    443 int bozo_init_httpd(bozohttpd_t *);
    444 int bozo_init_prefs(bozohttpd_t *, bozoprefs_t *);
    445 int bozo_set_defaults(bozohttpd_t *, bozoprefs_t *);
    446 int bozo_setup(bozohttpd_t *, bozoprefs_t *, const char *, const char *);
    447 void bozo_cleanup(bozohttpd_t *, bozoprefs_t *);
    448 bozo_httpreq_t *bozo_read_request(bozohttpd_t *);
    449 void bozo_process_request(bozo_httpreq_t *);
    450 void bozo_clean_request(bozo_httpreq_t *);
    451 int bozo_set_timeout(bozohttpd_t *, bozoprefs_t *, const char *, const char *);
    452 bozoheaders_t *addmerge_reqheader(bozo_httpreq_t *, const char *,
    453 				  const char *, ssize_t);
    454 bozoheaders_t *addmerge_replheader(bozo_httpreq_t *, const char *,
    455 				   const char *, ssize_t);
    456 
    457 /* variables */
    458 int bozo_set_pref(bozohttpd_t *, bozoprefs_t *, const char *, const char *);
    459 char *bozo_get_pref(bozoprefs_t *, const char *);
    460 
    461 int bozo_get_version(char */*buf*/, size_t /*size*/);
    462 
    463 extern volatile sig_atomic_t	bozo_timeout_hit;
    464 
    465 #endif	/* BOZOHTTPD_H_ */
    466