xref: /lighttpd1.4/src/http_header.c (revision e5f9e94d)
1c18f442aSGlenn Strauss /*
2c18f442aSGlenn Strauss  * http_header - HTTP header manipulation interfaces
3c18f442aSGlenn Strauss  *
4c18f442aSGlenn Strauss  * Copyright(c) 2018 Glenn Strauss gstrauss()gluelogic.com  All rights reserved
5c18f442aSGlenn Strauss  * License: BSD 3-clause (same as lighttpd)
6c18f442aSGlenn Strauss  */
73dd3cde9SGlenn Strauss #include "first.h"
83dd3cde9SGlenn Strauss 
94a196095SGlenn Strauss #include <string.h>
104a196095SGlenn Strauss #include "sys-strings.h"
114a196095SGlenn Strauss 
123dd3cde9SGlenn Strauss #include "http_header.h"
133dd3cde9SGlenn Strauss #include "array.h"
143dd3cde9SGlenn Strauss #include "buffer.h"
1567c0b149SGlenn Strauss #include "request.h"
163dd3cde9SGlenn Strauss 
173dd3cde9SGlenn Strauss 
183dd3cde9SGlenn Strauss typedef struct keyvlenvalue {
19367f30a6SGlenn Strauss     const  int16_t key;
20367f30a6SGlenn Strauss     const uint16_t vlen;
21367f30a6SGlenn Strauss     const char value[28];
223dd3cde9SGlenn Strauss } keyvlenvalue;
233dd3cde9SGlenn Strauss 
243dd3cde9SGlenn Strauss /* Note: must be sorted by length */
253dd3cde9SGlenn Strauss /* Note: must be kept in sync with http_header.h enum http_header_e */
26cff64cf1SGlenn Strauss /* Note: must be kept in sync http_headers[] and http_headers_off[] */
273fbb5773SGlenn Strauss /* Note: must be kept in sync h2.c:http_header_lc[] */
283fbb5773SGlenn Strauss /* Note: must be kept in sync h2.c:http_header_lshpack_idx[] */
293fbb5773SGlenn Strauss /* Note: must be kept in sync h2.c:lshpack_idx_http_header[] */
30cff64cf1SGlenn Strauss /* http_headers_off lists first offset at which string of specific len occur */
31367f30a6SGlenn Strauss static const int8_t http_headers_off[] = {
325a32a6dcSGlenn Strauss   -1, -1,  0,  1,  4,  9, 11, 17, 21, 26, 28, -1, 31, 32,
335a32a6dcSGlenn Strauss   38, 41, 46, 50, -1, 53, -1, -1, 54, 55, -1, 56, -1, 58
34cff64cf1SGlenn Strauss };
353dd3cde9SGlenn Strauss static const keyvlenvalue http_headers[] = {
36367f30a6SGlenn Strauss   { HTTP_HEADER_TE,                          CONST_LEN_STR("te") }
37367f30a6SGlenn Strauss  ,{ HTTP_HEADER_AGE,                         CONST_LEN_STR("age") }
38367f30a6SGlenn Strauss  ,{ HTTP_HEADER_DNT,                         CONST_LEN_STR("dnt") }
39367f30a6SGlenn Strauss  ,{ HTTP_HEADER_P3P,                         CONST_LEN_STR("p3p") }
40367f30a6SGlenn Strauss  ,{ HTTP_HEADER_HOST,                        CONST_LEN_STR("host") }
4192e5a021SGlenn Strauss  ,{ HTTP_HEADER_DATE,                        CONST_LEN_STR("date") }
4292e5a021SGlenn Strauss  ,{ HTTP_HEADER_ETAG,                        CONST_LEN_STR("etag") }
4392e5a021SGlenn Strauss  ,{ HTTP_HEADER_VARY,                        CONST_LEN_STR("vary") }
44367f30a6SGlenn Strauss  ,{ HTTP_HEADER_LINK,                        CONST_LEN_STR("link") }
45367f30a6SGlenn Strauss  ,{ HTTP_HEADER_ALLOW,                       CONST_LEN_STR("allow") }
4692e5a021SGlenn Strauss  ,{ HTTP_HEADER_RANGE,                       CONST_LEN_STR("range") }
4792e5a021SGlenn Strauss  ,{ HTTP_HEADER_COOKIE,                      CONST_LEN_STR("cookie") }
48367f30a6SGlenn Strauss  ,{ HTTP_HEADER_ACCEPT,                      CONST_LEN_STR("accept") }
4992e5a021SGlenn Strauss  ,{ HTTP_HEADER_STATUS,                      CONST_LEN_STR("status") }
5092e5a021SGlenn Strauss  ,{ HTTP_HEADER_SERVER,                      CONST_LEN_STR("server") }
51367f30a6SGlenn Strauss  ,{ HTTP_HEADER_EXPECT,                      CONST_LEN_STR("expect") }
52367f30a6SGlenn Strauss  ,{ HTTP_HEADER_PRAGMA,                      CONST_LEN_STR("pragma") }
5392e5a021SGlenn Strauss  ,{ HTTP_HEADER_UPGRADE,                     CONST_LEN_STR("upgrade") }
54367f30a6SGlenn Strauss  ,{ HTTP_HEADER_REFERER,                     CONST_LEN_STR("referer") }
55367f30a6SGlenn Strauss  ,{ HTTP_HEADER_EXPIRES,                     CONST_LEN_STR("expires") }
56367f30a6SGlenn Strauss  ,{ HTTP_HEADER_ALT_SVC,                     CONST_LEN_STR("alt-svc") }
5792e5a021SGlenn Strauss  ,{ HTTP_HEADER_LOCATION,                    CONST_LEN_STR("location") }
585a32a6dcSGlenn Strauss  ,{ HTTP_HEADER_PRIORITY,                    CONST_LEN_STR("priority") }
59367f30a6SGlenn Strauss  ,{ HTTP_HEADER_IF_MATCH,                    CONST_LEN_STR("if-match") }
60367f30a6SGlenn Strauss  ,{ HTTP_HEADER_IF_RANGE,                    CONST_LEN_STR("if-range") }
61367f30a6SGlenn Strauss  ,{ HTTP_HEADER_ALT_USED,                    CONST_LEN_STR("alt-used") }
6292e5a021SGlenn Strauss  ,{ HTTP_HEADER_FORWARDED,                   CONST_LEN_STR("forwarded") }
63367f30a6SGlenn Strauss  ,{ HTTP_HEADER_EXPECT_CT,                   CONST_LEN_STR("expect-ct") }
6492e5a021SGlenn Strauss  ,{ HTTP_HEADER_CONNECTION,                  CONST_LEN_STR("connection") }
6592e5a021SGlenn Strauss  ,{ HTTP_HEADER_SET_COOKIE,                  CONST_LEN_STR("set-cookie") }
6692e5a021SGlenn Strauss  ,{ HTTP_HEADER_USER_AGENT,                  CONST_LEN_STR("user-agent") }
6792e5a021SGlenn Strauss  ,{ HTTP_HEADER_CONTENT_TYPE,                CONST_LEN_STR("content-type") }
6892e5a021SGlenn Strauss  ,{ HTTP_HEADER_LAST_MODIFIED,               CONST_LEN_STR("last-modified") }
6992e5a021SGlenn Strauss  ,{ HTTP_HEADER_AUTHORIZATION,               CONST_LEN_STR("authorization") }
7092e5a021SGlenn Strauss  ,{ HTTP_HEADER_IF_NONE_MATCH,               CONST_LEN_STR("if-none-match") }
7192e5a021SGlenn Strauss  ,{ HTTP_HEADER_CACHE_CONTROL,               CONST_LEN_STR("cache-control") }
72367f30a6SGlenn Strauss  ,{ HTTP_HEADER_ACCEPT_RANGES,               CONST_LEN_STR("accept-ranges") }
73367f30a6SGlenn Strauss  ,{ HTTP_HEADER_CONTENT_RANGE,               CONST_LEN_STR("content-range") }
7492e5a021SGlenn Strauss  ,{ HTTP_HEADER_CONTENT_LENGTH,              CONST_LEN_STR("content-length") }
7592e5a021SGlenn Strauss  ,{ HTTP_HEADER_HTTP2_SETTINGS,              CONST_LEN_STR("http2-settings") }
76367f30a6SGlenn Strauss  ,{ HTTP_HEADER_ONION_LOCATION,              CONST_LEN_STR("onion-location") }
7792e5a021SGlenn Strauss  ,{ HTTP_HEADER_ACCEPT_ENCODING,             CONST_LEN_STR("accept-encoding") }
78367f30a6SGlenn Strauss  ,{ HTTP_HEADER_ACCEPT_LANGUAGE,             CONST_LEN_STR("accept-language") }
79367f30a6SGlenn Strauss  ,{ HTTP_HEADER_REFERRER_POLICY,             CONST_LEN_STR("referrer-policy") }
8092e5a021SGlenn Strauss  ,{ HTTP_HEADER_X_FORWARDED_FOR,             CONST_LEN_STR("x-forwarded-for") }
81367f30a6SGlenn Strauss  ,{ HTTP_HEADER_X_FRAME_OPTIONS,             CONST_LEN_STR("x-frame-options") }
82367f30a6SGlenn Strauss  ,{ HTTP_HEADER_WWW_AUTHENTICATE,            CONST_LEN_STR("www-authenticate") }
8392e5a021SGlenn Strauss  ,{ HTTP_HEADER_CONTENT_ENCODING,            CONST_LEN_STR("content-encoding") }
8492e5a021SGlenn Strauss  ,{ HTTP_HEADER_CONTENT_LOCATION,            CONST_LEN_STR("content-location") }
85367f30a6SGlenn Strauss  ,{ HTTP_HEADER_X_XSS_PROTECTION,            CONST_LEN_STR("x-xss-protection") }
8692e5a021SGlenn Strauss  ,{ HTTP_HEADER_IF_MODIFIED_SINCE,           CONST_LEN_STR("if-modified-since") }
8792e5a021SGlenn Strauss  ,{ HTTP_HEADER_TRANSFER_ENCODING,           CONST_LEN_STR("transfer-encoding") }
8892e5a021SGlenn Strauss  ,{ HTTP_HEADER_X_FORWARDED_PROTO,           CONST_LEN_STR("x-forwarded-proto") }
89367f30a6SGlenn Strauss  ,{ HTTP_HEADER_IF_UNMODIFIED_SINCE,         CONST_LEN_STR("if-unmodified-since") }
90367f30a6SGlenn Strauss  ,{ HTTP_HEADER_X_CONTENT_TYPE_OPTIONS,      CONST_LEN_STR("x-content-type-options") }
91367f30a6SGlenn Strauss  ,{ HTTP_HEADER_CONTENT_SECURITY_POLICY,     CONST_LEN_STR("content-security-policy") }
92367f30a6SGlenn Strauss  ,{ HTTP_HEADER_STRICT_TRANSPORT_SECURITY,   CONST_LEN_STR("strict-transport-security") }
93367f30a6SGlenn Strauss  ,{ HTTP_HEADER_UPGRADE_INSECURE_REQUESTS,   CONST_LEN_STR("upgrade-insecure-requests") }
94367f30a6SGlenn Strauss  ,{ HTTP_HEADER_ACCESS_CONTROL_ALLOW_ORIGIN, CONST_LEN_STR("access-control-allow-origin") }
95cff64cf1SGlenn Strauss  ,{ HTTP_HEADER_OTHER, 0, "" }
963dd3cde9SGlenn Strauss };
973dd3cde9SGlenn Strauss 
http_header_hkey_get(const char * const s,const size_t slen)98f8bd028dSGlenn Strauss enum http_header_e http_header_hkey_get(const char * const s, const size_t slen) {
99f8bd028dSGlenn Strauss     if (__builtin_expect( (slen < sizeof(http_headers_off)), 1)) {
100f8bd028dSGlenn Strauss         const int i = http_headers_off[slen];
101f8bd028dSGlenn Strauss         /*(lowercase first char as all recognized headers start w/ alpha char)*/
102f8bd028dSGlenn Strauss         const int c = s[0] | 0x20;
103f8bd028dSGlenn Strauss         const struct keyvlenvalue * restrict kv = http_headers + i;
104f8bd028dSGlenn Strauss         if (__builtin_expect( (i != -1), 1)) {
105b04f0311SGlenn Strauss             do {
106f8bd028dSGlenn Strauss                 if (__builtin_expect( (c != kv->value[0]), 1))
107f8bd028dSGlenn Strauss                     continue;
108f8bd028dSGlenn Strauss                 if (buffer_eq_icase_ssn(s+1, kv->value+1, slen-1))
109f8bd028dSGlenn Strauss                     return (enum http_header_e)kv->key;
110f8bd028dSGlenn Strauss             } while (slen == (++kv)->vlen);
111f8bd028dSGlenn Strauss         }
112f8bd028dSGlenn Strauss     }
1133dd3cde9SGlenn Strauss     return HTTP_HEADER_OTHER;
1143dd3cde9SGlenn Strauss }
1153dd3cde9SGlenn Strauss 
http_header_hkey_get_lc(const char * const s,const size_t slen)116f8bd028dSGlenn Strauss enum http_header_e http_header_hkey_get_lc(const char * const s, const size_t slen) {
117f8bd028dSGlenn Strauss     /* XXX: might not provide much real performance over http_header_hkey_get()
118*e5f9e94dSGlenn Strauss      *      (since the first-char comparison optimization was added)
119f8bd028dSGlenn Strauss      *      (and since well-known h2 headers are already mapped to hkey) */
120f8bd028dSGlenn Strauss     if (__builtin_expect( (slen < sizeof(http_headers_off)), 1)) {
121f8bd028dSGlenn Strauss         const int i = http_headers_off[slen];
122f8bd028dSGlenn Strauss         const int c = s[0];
123f8bd028dSGlenn Strauss         const struct keyvlenvalue * restrict kv = http_headers + i;
124f8bd028dSGlenn Strauss         if (__builtin_expect( (i != -1), 1)) {
12592e5a021SGlenn Strauss             do {
126f8bd028dSGlenn Strauss                 if (__builtin_expect( (c != kv->value[0]), 1))
127f8bd028dSGlenn Strauss                     continue;
128f8bd028dSGlenn Strauss                 if (0 == memcmp(s+1, kv->value+1, slen-1))
129f8bd028dSGlenn Strauss                     return (enum http_header_e)kv->key;
130f8bd028dSGlenn Strauss             } while (slen == (++kv)->vlen);
131f8bd028dSGlenn Strauss         }
132f8bd028dSGlenn Strauss     }
13392e5a021SGlenn Strauss     return HTTP_HEADER_OTHER;
13492e5a021SGlenn Strauss }
13592e5a021SGlenn Strauss 
1363dd3cde9SGlenn Strauss 
http_header_str_to_code(const char * const s)13710dbe38aSGlenn Strauss int http_header_str_to_code (const char * const s)
13810dbe38aSGlenn Strauss {
13910dbe38aSGlenn Strauss     /*(more strict than strtol(); exactly 3 digits followed by SP/TAB/NIL)*/
14010dbe38aSGlenn Strauss     return (light_isdigit(s[0]) && light_isdigit(s[1]) && light_isdigit(s[2])
14110dbe38aSGlenn Strauss             && (s[3] == '\0' || s[3] == ' ' || s[3] == '\t'))
14210dbe38aSGlenn Strauss       ? (s[0]-'0')*100 + (s[1]-'0')*10 + (s[2]-'0')
14310dbe38aSGlenn Strauss       : -1;
14410dbe38aSGlenn Strauss }
14510dbe38aSGlenn Strauss 
http_header_str_contains_token(const char * const s,const uint32_t slen,const char * const m,const uint32_t mlen)146a200e0c2SGlenn Strauss int http_header_str_contains_token (const char * const s, const uint32_t slen, const char * const m, const uint32_t mlen)
147e2b4c309SGlenn Strauss {
148e2b4c309SGlenn Strauss     /*if (slen < mlen) return 0;*//*(possible optimizations for caller)*/
149e2b4c309SGlenn Strauss     /*if (slen == mlen && buffer_eq_icase_ssn(s, m, mlen)) return 1;*/
150517e3278SGlenn Strauss     /*(note: does not handle quoted-string)*/
151a200e0c2SGlenn Strauss     uint32_t i = 0;
152e2b4c309SGlenn Strauss     do {
153e2b4c309SGlenn Strauss         while (i < slen &&  (s[i]==' ' || s[i]=='\t' || s[i]==',')) ++i;
1545547530aSGlenn Strauss         if (slen - i < mlen) return 0;
155e2b4c309SGlenn Strauss         if (buffer_eq_icase_ssn(s+i, m, mlen)) {
156e2b4c309SGlenn Strauss             i += mlen;
157e2b4c309SGlenn Strauss             if (i == slen || s[i]==' ' || s[i]=='\t' || s[i]==',' || s[i]==';')
158e2b4c309SGlenn Strauss                 return 1;
159e2b4c309SGlenn Strauss         }
160e2b4c309SGlenn Strauss         while (i < slen &&   s[i]!=',') ++i;
161e2b4c309SGlenn Strauss     } while (i < slen);
162e2b4c309SGlenn Strauss     return 0;
163e2b4c309SGlenn Strauss }
164e2b4c309SGlenn Strauss 
165785037ddSGlenn Strauss 
http_header_remove_token(buffer * const b,const char * const m,const uint32_t mlen)1660fb391c0SGlenn Strauss int http_header_remove_token (buffer * const b, const char * const m, const uint32_t mlen)
1670fb391c0SGlenn Strauss {
1680fb391c0SGlenn Strauss     /*(remove all instance of token from string)*/
1690fb391c0SGlenn Strauss     /*(note: does not handle quoted-string)*/
1700fb391c0SGlenn Strauss     int rc = 0;
1710fb391c0SGlenn Strauss     for (char *s = b->ptr; s; ) {
1720fb391c0SGlenn Strauss         while (*s == ' ' || *s == '\t' || *s == ',') ++s;
1730fb391c0SGlenn Strauss         if (0 == strncasecmp(s, m, mlen)) {
1740fb391c0SGlenn Strauss             s += mlen;
1750fb391c0SGlenn Strauss             if (*s=='\0' || *s==' ' || *s=='\t' || *s==',' || *s==';') {
1760fb391c0SGlenn Strauss                 memset(s-mlen, ' ', mlen);
1770fb391c0SGlenn Strauss                 while (*s != '\0' && *s != ',') ++s;
1780fb391c0SGlenn Strauss                 rc = 1;
1790fb391c0SGlenn Strauss                 if (*s == ',') {
1800fb391c0SGlenn Strauss                     *s++ = ' ';
1810fb391c0SGlenn Strauss                     continue;
1820fb391c0SGlenn Strauss                 }
1830fb391c0SGlenn Strauss                 else {
1840fb391c0SGlenn Strauss                     for (s -= mlen; *s != ',' && s != b->ptr; --s) ;
185af3df29aSGlenn Strauss                     buffer_truncate(b, (size_t)(s - b->ptr));
1860fb391c0SGlenn Strauss                     break;
1870fb391c0SGlenn Strauss                 }
1880fb391c0SGlenn Strauss             }
1890fb391c0SGlenn Strauss         }
1900fb391c0SGlenn Strauss         s = strchr(s, ',');
1910fb391c0SGlenn Strauss     }
1920fb391c0SGlenn Strauss     return rc;
1930fb391c0SGlenn Strauss }
1940fb391c0SGlenn Strauss 
1950fb391c0SGlenn Strauss 
http_header_token_append(buffer * const vb,const char * const v,const uint32_t vlen)196a200e0c2SGlenn Strauss static inline void http_header_token_append(buffer * const vb, const char * const v, const uint32_t vlen) {
197af3df29aSGlenn Strauss     if (!buffer_is_blank(vb))
198785037ddSGlenn Strauss         buffer_append_string_len(vb, CONST_STR_LEN(", "));
199785037ddSGlenn Strauss     buffer_append_string_len(vb, v, vlen);
200785037ddSGlenn Strauss }
201785037ddSGlenn Strauss 
20205ff9c57SGlenn Strauss __attribute_cold__
http_header_token_append_cookie(buffer * const vb,const char * const v,const uint32_t vlen)20305ff9c57SGlenn Strauss static inline void http_header_token_append_cookie(buffer * const vb, const char * const v, const uint32_t vlen) {
20405ff9c57SGlenn Strauss     /* Cookie request header must be special-cased to use ';' separator
20505ff9c57SGlenn Strauss      * instead of ',' to combine multiple headers (if present) */
206af3df29aSGlenn Strauss     if (!buffer_is_blank(vb))
20705ff9c57SGlenn Strauss         buffer_append_string_len(vb, CONST_STR_LEN("; "));
20805ff9c57SGlenn Strauss     buffer_append_string_len(vb, v, vlen);
20905ff9c57SGlenn Strauss }
21005ff9c57SGlenn Strauss 
211785037ddSGlenn Strauss __attribute_pure__
http_header_generic_get_ifnotempty(const array * const a,const enum http_header_e id,const char * const k,const uint32_t klen)2122e0676fdSGlenn Strauss static inline buffer * http_header_generic_get_ifnotempty(const array * const a, const enum http_header_e id, const char * const k, const uint32_t klen) {
213b7942c58SGlenn Strauss     data_string * const ds =
2142e0676fdSGlenn Strauss       (data_string *)array_get_element_klen_ext(a, id, k, klen);
215af3df29aSGlenn Strauss     return ds && !buffer_is_blank(&ds->value) ? &ds->value : NULL;
2163dd3cde9SGlenn Strauss }
2173dd3cde9SGlenn Strauss 
http_header_set_key_value(array * const a,enum http_header_e id,const char * const k,const size_t klen,const char * const v,const size_t vlen)2182e0676fdSGlenn Strauss static inline void http_header_set_key_value(array * const a, enum http_header_e id, const char * const k, const size_t klen, const char * const v, const size_t vlen) {
2192e0676fdSGlenn Strauss     buffer_copy_string_len(array_get_buf_ptr_ext(a, id, k, klen), v, vlen);
2202e0676fdSGlenn Strauss }
2212e0676fdSGlenn Strauss 
222785037ddSGlenn Strauss 
http_header_response_get(const request_st * const r,enum http_header_e id,const char * k,uint32_t klen)223a200e0c2SGlenn Strauss buffer * http_header_response_get(const request_st * const r, enum http_header_e id, const char *k, uint32_t klen) {
2242e0676fdSGlenn Strauss     return light_btst(r->resp_htags, id)
2252e0676fdSGlenn Strauss       ? http_header_generic_get_ifnotempty(&r->resp_headers, id, k, klen)
226785037ddSGlenn Strauss       : NULL;
227785037ddSGlenn Strauss }
228785037ddSGlenn Strauss 
http_header_response_set_ptr(request_st * const r,enum http_header_e id,const char * k,uint32_t klen)22926f354cbSGlenn Strauss buffer * http_header_response_set_ptr(request_st * const r, enum http_header_e id, const char *k, uint32_t klen) {
23026f354cbSGlenn Strauss     /* note: caller must not leave buffer empty
23126f354cbSGlenn Strauss      * or must call http_header_response_unset() */
23226f354cbSGlenn Strauss     light_bset(r->resp_htags, id);
23326f354cbSGlenn Strauss     buffer * const vb = array_get_buf_ptr_ext(&r->resp_headers, id, k, klen);
23426f354cbSGlenn Strauss     buffer_clear(vb);
23526f354cbSGlenn Strauss     return vb;
23626f354cbSGlenn Strauss }
23726f354cbSGlenn Strauss 
http_header_response_unset(request_st * const r,enum http_header_e id,const char * k,uint32_t klen)238a200e0c2SGlenn Strauss void http_header_response_unset(request_st * const r, enum http_header_e id, const char *k, uint32_t klen) {
2392e0676fdSGlenn Strauss     if (light_btst(r->resp_htags, id)) {
2402e0676fdSGlenn Strauss         /* (do not clear bit for HTTP_HEADER_OTHER,
2412e0676fdSGlenn Strauss          *  as there might be addtl "other" headers) */
2429c8981a7SGlenn Strauss         if (id > HTTP_HEADER_OTHER) light_bclr(r->resp_htags, id);
2432e0676fdSGlenn Strauss         http_header_set_key_value(&r->resp_headers,id,k,klen,CONST_STR_LEN(""));
244f13db690SGlenn Strauss     }
245f13db690SGlenn Strauss }
246f13db690SGlenn Strauss 
http_header_response_set(request_st * const r,enum http_header_e id,const char * k,uint32_t klen,const char * v,uint32_t vlen)247a200e0c2SGlenn Strauss void http_header_response_set(request_st * const r, enum http_header_e id, const char *k, uint32_t klen, const char *v, uint32_t vlen) {
2483dd3cde9SGlenn Strauss     /* set value, including setting blank value if 0 == vlen
2493dd3cde9SGlenn Strauss      * (note: if 0 == vlen, header is still inserted with blank value,
2503dd3cde9SGlenn Strauss      *  which is used to indicate a "removed" header)
2512e0676fdSGlenn Strauss      * (do not clear bit for HTTP_HEADER_OTHER if 0 == vlen,
2522e0676fdSGlenn Strauss      *  as there might be addtl "other" headers) */
2532e0676fdSGlenn Strauss     (vlen)
2542e0676fdSGlenn Strauss       ? light_bset(r->resp_htags, id)
2552e0676fdSGlenn Strauss       : (id > HTTP_HEADER_OTHER ? light_bclr(r->resp_htags, id) : 0);
2562e0676fdSGlenn Strauss     http_header_set_key_value(&r->resp_headers, id, k, klen, v, vlen);
2573dd3cde9SGlenn Strauss }
2583dd3cde9SGlenn Strauss 
http_header_response_append(request_st * const r,enum http_header_e id,const char * k,uint32_t klen,const char * v,uint32_t vlen)259a200e0c2SGlenn Strauss void http_header_response_append(request_st * const r, enum http_header_e id, const char *k, uint32_t klen, const char *v, uint32_t vlen) {
260b2991c68SGlenn Strauss     if (0 == vlen) return;
2612e0676fdSGlenn Strauss     light_bset(r->resp_htags, id);
2622e0676fdSGlenn Strauss     buffer * const vb = array_get_buf_ptr_ext(&r->resp_headers, id, k, klen);
263785037ddSGlenn Strauss     http_header_token_append(vb, v, vlen);
2643dd3cde9SGlenn Strauss }
2653dd3cde9SGlenn Strauss 
2662e0676fdSGlenn Strauss __attribute_cold__
http_header_response_insert_addtl(request_st * const r,enum http_header_e id,const char * k,uint32_t klen,buffer * const vb,uint32_t vlen)2672e0676fdSGlenn Strauss static void http_header_response_insert_addtl(request_st * const r, enum http_header_e id, const char *k, uint32_t klen, buffer * const vb, uint32_t vlen) {
2682e0676fdSGlenn Strauss     UNUSED(id);
269dc01487eSGlenn Strauss     char *h = buffer_string_prepare_append(vb, 2 + klen + vlen + 2);
270dc01487eSGlenn Strauss     buffer_append_str3(vb, CONST_STR_LEN("\r\n"), k, klen, CONST_STR_LEN(": "));
27177057a7cSGlenn Strauss     if (r->http_version >= HTTP_VERSION_2) {
272ada09a23SGlenn Strauss         r->resp_header_repeated = 1;
273dc01487eSGlenn Strauss         h += 2;
274dc01487eSGlenn Strauss         for (uint32_t i = 0; i < klen; ++i) {
275dc01487eSGlenn Strauss             if (light_isupper(h[i])) h[i] |= 0x20;
27677057a7cSGlenn Strauss         }
277dc01487eSGlenn Strauss     }
278f13db690SGlenn Strauss }
2792e0676fdSGlenn Strauss 
http_header_response_insert(request_st * const r,enum http_header_e id,const char * k,uint32_t klen,const char * v,uint32_t vlen)2802e0676fdSGlenn Strauss void http_header_response_insert(request_st * const r, enum http_header_e id, const char *k, uint32_t klen, const char *v, uint32_t vlen) {
2812e0676fdSGlenn Strauss     if (0 == vlen) return;
2822e0676fdSGlenn Strauss     light_bset(r->resp_htags, id);
2832e0676fdSGlenn Strauss     buffer * const vb = array_get_buf_ptr_ext(&r->resp_headers, id, k, klen);
284af3df29aSGlenn Strauss     if (!buffer_is_blank(vb)) /*append repeated field-name on new line*/
2852e0676fdSGlenn Strauss         http_header_response_insert_addtl(r, id, k, klen, vb, vlen);
2863dd3cde9SGlenn Strauss     buffer_append_string_len(vb, v, vlen);
2873dd3cde9SGlenn Strauss }
2883dd3cde9SGlenn Strauss 
2893dd3cde9SGlenn Strauss 
http_header_request_get(const request_st * const r,enum http_header_e id,const char * k,uint32_t klen)290a200e0c2SGlenn Strauss buffer * http_header_request_get(const request_st * const r, enum http_header_e id, const char *k, uint32_t klen) {
2912e0676fdSGlenn Strauss     return light_btst(r->rqst_htags, id)
2922e0676fdSGlenn Strauss       ? http_header_generic_get_ifnotempty(&r->rqst_headers, id, k, klen)
2933dd3cde9SGlenn Strauss       : NULL;
2943dd3cde9SGlenn Strauss }
2953dd3cde9SGlenn Strauss 
http_header_request_set_ptr(request_st * const r,enum http_header_e id,const char * k,uint32_t klen)29626f354cbSGlenn Strauss buffer * http_header_request_set_ptr(request_st * const r, enum http_header_e id, const char *k, uint32_t klen) {
29726f354cbSGlenn Strauss     /* note: caller must not leave buffer empty
29826f354cbSGlenn Strauss      * or must call http_header_request_unset() */
29926f354cbSGlenn Strauss     light_bset(r->rqst_htags, id);
30026f354cbSGlenn Strauss     buffer * const vb = array_get_buf_ptr_ext(&r->rqst_headers, id, k, klen);
30126f354cbSGlenn Strauss     buffer_clear(vb);
30226f354cbSGlenn Strauss     return vb;
30326f354cbSGlenn Strauss }
30426f354cbSGlenn Strauss 
http_header_request_unset(request_st * const r,enum http_header_e id,const char * k,uint32_t klen)305a200e0c2SGlenn Strauss void http_header_request_unset(request_st * const r, enum http_header_e id, const char *k, uint32_t klen) {
3062e0676fdSGlenn Strauss     if (light_btst(r->rqst_htags, id)) {
3072e0676fdSGlenn Strauss         /* (do not clear bit for HTTP_HEADER_OTHER,
3082e0676fdSGlenn Strauss          *  as there might be addtl "other" headers) */
3099c8981a7SGlenn Strauss         if (id > HTTP_HEADER_OTHER) light_bclr(r->rqst_htags, id);
3102e0676fdSGlenn Strauss         http_header_set_key_value(&r->rqst_headers,id,k,klen,CONST_STR_LEN(""));
311f13db690SGlenn Strauss     }
312f13db690SGlenn Strauss }
313f13db690SGlenn Strauss 
http_header_request_set(request_st * const r,enum http_header_e id,const char * k,uint32_t klen,const char * v,uint32_t vlen)314a200e0c2SGlenn Strauss void http_header_request_set(request_st * const r, enum http_header_e id, const char *k, uint32_t klen, const char *v, uint32_t vlen) {
3153dd3cde9SGlenn Strauss     /* set value, including setting blank value if 0 == vlen
3163dd3cde9SGlenn Strauss      * (note: if 0 == vlen, header is still inserted with blank value,
3173dd3cde9SGlenn Strauss      *  which is used to indicate a "removed" header)
3182e0676fdSGlenn Strauss      * (do not clear bit for HTTP_HEADER_OTHER if 0 == vlen,
3192e0676fdSGlenn Strauss      *  as there might be addtl "other" headers) */
3202e0676fdSGlenn Strauss     (vlen)
3212e0676fdSGlenn Strauss       ? light_bset(r->rqst_htags, id)
3222e0676fdSGlenn Strauss       : (id > HTTP_HEADER_OTHER ? light_bclr(r->rqst_htags, id) : 0);
3232e0676fdSGlenn Strauss     http_header_set_key_value(&r->rqst_headers, id, k, klen, v, vlen);
3243dd3cde9SGlenn Strauss }
3253dd3cde9SGlenn Strauss 
http_header_request_append(request_st * const r,enum http_header_e id,const char * k,uint32_t klen,const char * v,uint32_t vlen)326a200e0c2SGlenn Strauss void http_header_request_append(request_st * const r, enum http_header_e id, const char *k, uint32_t klen, const char *v, uint32_t vlen) {
327b2991c68SGlenn Strauss     if (0 == vlen) return;
3282e0676fdSGlenn Strauss     light_bset(r->rqst_htags, id);
3292e0676fdSGlenn Strauss     buffer * const vb = array_get_buf_ptr_ext(&r->rqst_headers, id, k, klen);
33005ff9c57SGlenn Strauss     if (id != HTTP_HEADER_COOKIE)
331785037ddSGlenn Strauss         http_header_token_append(vb, v, vlen);
33205ff9c57SGlenn Strauss     else
33305ff9c57SGlenn Strauss         http_header_token_append_cookie(vb, v, vlen);
3343dd3cde9SGlenn Strauss }
3353dd3cde9SGlenn Strauss 
3363dd3cde9SGlenn Strauss 
http_header_env_get(const request_st * const r,const char * k,uint32_t klen)337a200e0c2SGlenn Strauss buffer * http_header_env_get(const request_st * const r, const char *k, uint32_t klen) {
3382e0676fdSGlenn Strauss     /* similar to http_header_generic_get_ifnotempty() but without id */
3392e0676fdSGlenn Strauss     data_string * const ds =
3402e0676fdSGlenn Strauss       (data_string *)array_get_element_klen(&r->env, k, klen);
341af3df29aSGlenn Strauss     return ds && !buffer_is_blank(&ds->value) ? &ds->value : NULL;
3423dd3cde9SGlenn Strauss }
3433dd3cde9SGlenn Strauss 
http_header_env_set_ptr(request_st * r,const char * k,uint32_t klen)34426f354cbSGlenn Strauss buffer * http_header_env_set_ptr(request_st *r, const char *k, uint32_t klen) {
34526f354cbSGlenn Strauss     buffer * const vb = array_get_buf_ptr(&r->env, k, klen);
34626f354cbSGlenn Strauss     buffer_clear(vb);
34726f354cbSGlenn Strauss     return vb;
34826f354cbSGlenn Strauss }
34926f354cbSGlenn Strauss 
http_header_env_set(request_st * const r,const char * k,uint32_t klen,const char * v,uint32_t vlen)350a200e0c2SGlenn Strauss void http_header_env_set(request_st * const r, const char *k, uint32_t klen, const char *v, uint32_t vlen) {
3517c7f8c46SGlenn Strauss     array_set_key_value(&r->env, k, klen, v, vlen);
3523dd3cde9SGlenn Strauss }
3533dd3cde9SGlenn Strauss 
http_header_env_append(request_st * const r,const char * k,uint32_t klen,const char * v,uint32_t vlen)354a200e0c2SGlenn Strauss void http_header_env_append(request_st * const r, const char *k, uint32_t klen, const char *v, uint32_t vlen) {
355b2991c68SGlenn Strauss     /*if (0 == vlen) return;*//* skip check; permit env var w/ blank value */
3567c7f8c46SGlenn Strauss     buffer * const vb = array_get_buf_ptr(&r->env, k, klen);
357785037ddSGlenn Strauss     http_header_token_append(vb, v, vlen);
3583dd3cde9SGlenn Strauss }
359db7b51a4SGlenn Strauss 
360db7b51a4SGlenn Strauss 
361db7b51a4SGlenn Strauss uint32_t
http_header_parse_hoff(const char * n,const uint32_t clen,unsigned short hoff[8192])362db7b51a4SGlenn Strauss http_header_parse_hoff (const char *n, const uint32_t clen, unsigned short hoff[8192])
363db7b51a4SGlenn Strauss {
364db7b51a4SGlenn Strauss     uint32_t hlen = 0;
365db7b51a4SGlenn Strauss     for (const char *b; (n = memchr((b = n),'\n',clen-hlen)); ++n) {
366db7b51a4SGlenn Strauss         uint32_t x = (uint32_t)(n - b + 1);
367db7b51a4SGlenn Strauss         hlen += x;
368db7b51a4SGlenn Strauss         if (x <= 2 && (x == 1 || n[-1] == '\r')) {
369db7b51a4SGlenn Strauss             hoff[hoff[0]+1] = hlen;
370db7b51a4SGlenn Strauss             return hlen;
371db7b51a4SGlenn Strauss         }
372db7b51a4SGlenn Strauss         if (++hoff[0] >= /*sizeof(hoff)/sizeof(hoff[0])-1*/ 8192-1) break;
373db7b51a4SGlenn Strauss         hoff[hoff[0]] = hlen;
374db7b51a4SGlenn Strauss     }
375db7b51a4SGlenn Strauss     return 0;
376db7b51a4SGlenn Strauss }
377