#ifndef HTTPD_H #define HTTPD_H #include #include #include #include #include "evnt.h" #define HTTPD_CONF_SERV_DEF_PORT 80 /* if the data is less than this, queue instead of hitting write */ #define HTTPD_CONF_MAX_WAIT_SEND 16 #define HTTPD_CONF_RECV_CALL_LIMIT 3 /* read+send+sendfile */ #define HTTPD_CONF_SEND_CALL_LIMIT 1 /* already blocking in send */ struct Con; struct Httpd_req_data; #include "httpd_conf_main.h" #include "httpd_conf_req.h" #include "httpd_err_codes.h" struct Http_hdrs { Vstr_sect_node hdr_ua[1]; Vstr_sect_node hdr_referer[1]; /* NOTE: referrer */ Vstr_sect_node hdr_expect[1]; Vstr_sect_node hdr_host[1]; Vstr_sect_node hdr_if_modified_since[1]; Vstr_sect_node hdr_if_range[1]; Vstr_sect_node hdr_if_unmodified_since[1]; Vstr_sect_node hdr_authorization[1]; /* can have multiple headers... */ struct Http_hdrs__multi { Vstr_base *combiner_store; Vstr_base *comb; Vstr_sect_node hdr_accept[1]; Vstr_sect_node hdr_accept_charset[1]; Vstr_sect_node hdr_accept_encoding[1]; Vstr_sect_node hdr_accept_language[1]; Vstr_sect_node hdr_connection[1]; Vstr_sect_node hdr_if_match[1]; Vstr_sect_node hdr_if_none_match[1]; Vstr_sect_node hdr_range[1]; } multi[1]; }; #define HTTPD__DECL_XTRA_HDR(x) \ const Vstr_base * x ## _vs1; \ size_t x ## _pos; \ size_t x ## _len typedef struct Httpd_req_data { const Httpd_policy_opts *policy; struct Http_hdrs http_hdrs[1]; Vstr_base *fname; size_t len; size_t path_pos; size_t path_len; unsigned int error_code; const char *error_line; const char *error_msg; VSTR_AUTOCONF_uintmax_t error_len; /* due to custom files */ Vstr_sects *sects; struct stat64 f_stat[1]; size_t orig_io_w_len; Vstr_base *f_mmap; Vstr_base *xtra_content; HTTPD__DECL_XTRA_HDR(content_type); HTTPD__DECL_XTRA_HDR(content_language); HTTPD__DECL_XTRA_HDR(content_location); HTTPD__DECL_XTRA_HDR(content_md5); /* Note this is valid for range */ time_t content_md5_time; HTTPD__DECL_XTRA_HDR(gzip_content_md5); time_t gzip_content_md5_time; HTTPD__DECL_XTRA_HDR(bzip2_content_md5); time_t bzip2_content_md5_time; HTTPD__DECL_XTRA_HDR(cache_control); HTTPD__DECL_XTRA_HDR(etag); time_t etag_time; HTTPD__DECL_XTRA_HDR(gzip_etag); time_t gzip_etag_time; HTTPD__DECL_XTRA_HDR(bzip2_etag); time_t bzip2_etag_time; HTTPD__DECL_XTRA_HDR(expires); time_t expires_time; HTTPD__DECL_XTRA_HDR(link); HTTPD__DECL_XTRA_HDR(p3p); HTTPD__DECL_XTRA_HDR(ext_vary_a); HTTPD__DECL_XTRA_HDR(ext_vary_ac); HTTPD__DECL_XTRA_HDR(ext_vary_al); time_t encoded_mtime; time_t now; size_t vhost_prefix_len; unsigned int ver_0_9 : 1; unsigned int ver_1_1 : 1; unsigned int head_op : 1; unsigned int content_encoding_gzip : 1; unsigned int content_encoding_xgzip : 1; /* only valid if gzip is TRUE */ unsigned int content_encoding_bzip2 : 1; unsigned int content_encoding_identity : 1; unsigned int direct_uri : 1; unsigned int direct_filename : 1; unsigned int skip_document_root : 1; unsigned int parse_accept : 1; unsigned int parse_accept_language : 1; unsigned int allow_accept_encoding : 1; unsigned int output_keep_alive_hdr : 1; unsigned int user_return_error_code : 1; unsigned int vary_star : 1; unsigned int vary_a : 1; unsigned int vary_ac : 1; unsigned int vary_ae : 1; unsigned int vary_al : 1; unsigned int vary_rf : 1; unsigned int vary_ua : 1; unsigned int chked_encoded_path : 1; unsigned int chk_encoded_slash : 1; unsigned int chk_encoded_dot : 1; unsigned int neg_content_type_done : 1; unsigned int neg_content_lang_done : 1; unsigned int using_req : 1; unsigned int done_once : 1; unsigned int malloc_bad : 1; } Httpd_req_data; struct File_sect { int fd; VSTR_AUTOCONF_uintmax_t off; VSTR_AUTOCONF_uintmax_t len; }; struct Con { struct Evnt evnt[1]; Vstr_ref *acpt_sa_ref; const Httpd_policy_opts *policy; Vstr_base *mpbr_ct; /* multipart/byterange */ VSTR_AUTOCONF_uintmax_t mpbr_fs_len; unsigned int fs_off; unsigned int fs_num; unsigned int fs_sz; struct File_sect *fs; struct File_sect fs_store[1]; unsigned int io_limit_num : 8; unsigned int vary_star : 1; unsigned int parsed_method_ver_1_0 : 1; unsigned int keep_alive : 3; unsigned int use_mmap : 1; unsigned int use_sendfile : 1; unsigned int use_posix_fadvise : 1; unsigned int use_mpbr : 1; }; #define CON_CEVNT_SA(x) EVNT_SA((x)->evnt) #define CON_CEVNT_SA_IN4(x) EVNT_SA_IN4((x)->evnt) #define CON_CEVNT_SA_IN6(x) EVNT_SA_IN6((x)->evnt) #define CON_CEVNT_SA_UN(x) EVNT_SA_UN((x)->evnt) #define CON_SEVNT_SA(x) ACPT_SA((Acpt_data *)((x)->acpt_sa_ref->ptr)) #define CON_SEVNT_SA_IN4(x) ACPT_SA_IN4((Acpt_data *)((x)->acpt_sa_ref->ptr)) #define CON_SEVNT_SA_IN6(x) ACPT_SA_IN6((Acpt_data *)((x)->acpt_sa_ref->ptr)) #define CON_SEVNT_SA_UN(x) ACPT_SA_UN((Acpt_data *)((x)->acpt_sa_ref->ptr)) #define HTTP_NON_KEEP_ALIVE 0 #define HTTP_1_0_KEEP_ALIVE 1 #define HTTP_1_1_KEEP_ALIVE 2 /* Linear Whitespace, a full stds. check for " " and "\t" */ #define HTTP_LWS " \t" /* End of Line */ #define HTTP_EOL "\r\n" /* End of Request - blank line following previous line in request */ #define HTTP_END_OF_REQUEST HTTP_EOL HTTP_EOL /* HTTP crack -- Implied linear whitespace between tokens, note that it * is *LWS == *([CRLF] 1*(SP | HT)) */ #define HTTP_SKIP_LWS(s1, p, l) do { \ char http__q_tst_lws = 0; \ \ if (!l) break; \ http__q_tst_lws = vstr_export_chr(s1, p); \ if ((http__q_tst_lws != '\r') && (http__q_tst_lws != ' ') && \ (http__q_tst_lws != '\t')) \ break; \ \ http__skip_lws(s1, &p, &l); \ } while (FALSE) #define HTTPD_APP_REF_VSTR(x, s2, p2, l2) \ vstr_add_vstr(x, (x)->len, s2, p2, l2, VSTR_TYPE_ADD_BUF_REF) #define HTTPD_APP_REF_ALLVSTR(x, s2) \ vstr_add_vstr(x, (x)->len, s2, 1, (s2)->len, VSTR_TYPE_ADD_BUF_REF) #ifdef HTTPD_HAVE_GLOBAL_OPTS extern Httpd_opts httpd_opts[1]; /* hacky ... */ #endif extern void httpd_init(Vlg *); extern void httpd_exit(void); extern Httpd_req_data *http_req_make(struct Con *); extern void http_req_free(Httpd_req_data *); extern void http_req_exit(void); extern void httpd_fin_fd_close(struct Con *); extern int httpd_valid_url_filename(Vstr_base *, size_t, size_t); extern int httpd_init_default_hostname(Opt_serv_policy_opts *); extern int httpd_sc_add_default_hostname(struct Con *, Httpd_req_data *, Vstr_base *, size_t); extern int httpd_canon_path(Vstr_base *); extern int httpd_canon_dir_path(Vstr_base *); extern int httpd_canon_abs_dir_path(Vstr_base *); extern void httpd_req_absolute_uri(struct Con *, Httpd_req_data *, Vstr_base *, size_t, size_t); extern int http_req_chk_dir(struct Con *, Httpd_req_data *); extern int http_req_content_type(Httpd_req_data *); extern unsigned int http_parse_accept(Httpd_req_data *, const Vstr_base *, size_t, size_t); extern unsigned int http_parse_accept_language(Httpd_req_data *, const Vstr_base *, size_t, size_t); extern void http_app_def_hdrs(struct Con *, Httpd_req_data *, unsigned int, const char *, time_t, const char *, int, VSTR_AUTOCONF_uintmax_t); extern int http_fmt_add_vstr_add_vstr(Vstr_conf *, const char *); extern int http_fmt_add_vstr_add_sect_vstr(Vstr_conf *, const char *); extern int http_req_op_get(struct Con *, Httpd_req_data *); extern int http_req_op_opts(struct Con *, Httpd_req_data *); extern int http_req_op_trace(struct Con *, Httpd_req_data *); extern int httpd_serv_send(struct Con *); extern int httpd_serv_recv(struct Con *); extern int httpd_con_init(struct Con *, struct Acpt_listener *); #endif