varnish-cache/bin/varnishd/cache/cache_http.c
0
/*-
1
 * Copyright (c) 2006 Verdens Gang AS
2
 * Copyright (c) 2006-2017 Varnish Software AS
3
 * All rights reserved.
4
 *
5
 * Author: Poul-Henning Kamp <phk@phk.freebsd.dk>
6
 *
7
 * SPDX-License-Identifier: BSD-2-Clause
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 in the
16
 *    documentation and/or other materials provided with the distribution.
17
 *
18
 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
19
 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
20
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
21
 * ARE DISCLAIMED.  IN NO EVENT SHALL AUTHOR OR CONTRIBUTORS BE LIABLE
22
 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
23
 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
24
 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
25
 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
26
 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
27
 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
28
 * SUCH DAMAGE.
29
 *
30
 * HTTP request storage and manipulation
31
 */
32
33
#include "config.h"
34
35
#include "cache_varnishd.h"
36
#include <stdio.h>
37
#include <stdlib.h>
38
39
#include "common/heritage.h"
40
41
#include "vct.h"
42
#include "vend.h"
43
#include "vnum.h"
44
#include "vtim.h"
45
46
#define BODYSTATUS(U, l, n, a, k)                               \
47
        const struct body_status BS_##U[1] = {{                 \
48
                .name = #l,                                     \
49
                .nbr = n,                                       \
50
                .avail = a,                                     \
51
                .length_known = k                               \
52
        }};
53
#include "tbl/body_status.h"
54
55
56
#define HTTPH(a, b, c) hdr_t b = HDR(a);
57
#include "tbl/http_headers.h"
58
59
hdr_t H__Status = HDR(":status");
60
hdr_t H__Proto  = HDR(":proto");
61
hdr_t H__Reason = HDR(":reason");
62
63
static char * via_hdr;
64
65
/*--------------------------------------------------------------------
66
 * Perfect hash to rapidly recognize headers from tbl/http_headers.h
67
 * which have non-zero flags.
68
 *
69
 * A suitable algorithm can be found with `gperf`:
70
 *
71
 *      tr '" ,' '   ' < include/tbl/http_headers.h |
72
 *              awk '$1 == "H(" {print $2}' |
73
 *              gperf --ignore-case
74
 *
75
 */
76
77
#define GPERF_MIN_WORD_LENGTH 2
78
#define GPERF_MAX_WORD_LENGTH 19
79
#define GPERF_MAX_HASH_VALUE 79
80
81
static const unsigned char http_asso_values[256] = {
82
        80, 80, 80, 80, 80, 80, 80, 80, 80, 80,
83
        80, 80, 80, 80, 80, 80, 80, 80, 80, 80,
84
        80, 80, 80, 80, 80, 80, 80, 80, 80, 80,
85
        80, 80, 80, 80, 80, 80, 80, 80, 80, 80,
86
        80, 80, 80, 80, 80, 80, 80, 80, 80, 80,
87
        80, 80, 80,  0, 80, 80, 80, 80, 80, 80,
88
        80, 80, 80, 80, 80,  5, 80, 20,  0,  0,
89
        5, 10,  5,  5, 80,  0, 15,  0, 20, 80,
90
        40, 80,  0, 35, 10, 20, 55, 45,  0,  0,
91
        80, 80, 80, 80, 80, 80, 80,  5, 80, 20,
92
        0,  0,  5, 10,  5,  5, 80,  0, 15,  0,
93
        20, 80, 40, 80,  0, 35, 10, 20, 55, 45,
94
        0,  0, 80, 80, 80, 80, 80, 80, 80, 80,
95
        80, 80, 80, 80, 80, 80, 80, 80, 80, 80,
96
        80, 80, 80, 80, 80, 80, 80, 80, 80, 80,
97
        80, 80, 80, 80, 80, 80, 80, 80, 80, 80,
98
        80, 80, 80, 80, 80, 80, 80, 80, 80, 80,
99
        80, 80, 80, 80, 80, 80, 80, 80, 80, 80,
100
        80, 80, 80, 80, 80, 80, 80, 80, 80, 80,
101
        80, 80, 80, 80, 80, 80, 80, 80, 80, 80,
102
        80, 80, 80, 80, 80, 80, 80, 80, 80, 80,
103
        80, 80, 80, 80, 80, 80, 80, 80, 80, 80,
104
        80, 80, 80, 80, 80, 80, 80, 80, 80, 80,
105
        80, 80, 80, 80, 80, 80, 80, 80, 80, 80,
106
        80, 80, 80, 80, 80, 80, 80, 80, 80, 80,
107
        80, 80, 80, 80, 80, 80
108
};
109
110
static struct http_hdrflg {
111
        hdr_t           *hdr;
112
        unsigned        flag;
113
} http_hdrflg[GPERF_MAX_HASH_VALUE + 1] = {
114
        { NULL }, { NULL }, { NULL }, { NULL },
115
        { &H_Date },
116
        { &H_Range },
117
        { NULL },
118
        { &H_Referer },
119
        { &H_Age },
120
        { &H_From },
121
        { &H_Keep_Alive },
122
        { &H_Retry_After },
123
        { &H_TE },
124
        { &H_If_Range },
125
        { &H_ETag },
126
        { &H_X_Forwarded_For },
127
        { &H_Expect },
128
        { &H_Trailer },
129
        { &H_If_Match },
130
        { &H_Host },
131
        { &H_Accept_Language },
132
        { &H_Accept },
133
        { &H_If_Modified_Since },
134
        { &H_If_None_Match },
135
        { &H_If_Unmodified_Since },
136
        { NULL },
137
        { &H_Cookie },
138
        { &H_Upgrade },
139
        { &H_Last_Modified },
140
        { &H_Accept_Charset },
141
        { &H_Accept_Encoding },
142
        { &H_Content_MD5 },
143
        { &H_Content_Type },
144
        { &H_Content_Range },
145
        { NULL }, { NULL },
146
        { &H_Content_Language },
147
        { &H_Transfer_Encoding },
148
        { &H_Authorization },
149
        { &H_Content_Length },
150
        { &H_User_Agent },
151
        { &H_Server },
152
        { &H_Expires },
153
        { &H_Location },
154
        { NULL },
155
        { &H_Set_Cookie },
156
        { &H_Content_Encoding },
157
        { &H_Max_Forwards },
158
        { &H_Cache_Control },
159
        { NULL },
160
        { &H_Connection },
161
        { &H_Pragma },
162
        { NULL },
163
        { &H_Accept_Ranges },
164
        { &H_HTTP2_Settings },
165
        { &H_Allow },
166
        { &H_Content_Location },
167
        { NULL },
168
        { &H_Proxy_Authenticate },
169
        { &H_Vary },
170
        { NULL },
171
        { &H_WWW_Authenticate },
172
        { &H_Warning },
173
        { &H_Via },
174
        { NULL }, { NULL }, { NULL }, { NULL },
175
        { NULL }, { NULL }, { NULL }, { NULL },
176
        { NULL }, { NULL }, { NULL }, { NULL },
177
        { NULL }, { NULL }, { NULL },
178
        { &H_Proxy_Authorization }
179
};
180
181
static struct http_hdrflg *
182 75370
http_hdr_flags(const char *b, const char *e)
183
{
184
        unsigned u;
185
        struct http_hdrflg *retval;
186
187 75370
        if (b == NULL || e == NULL)
188 4
                return (NULL);
189 75366
        u = pdiff(b, e);
190 75366
        if (u < GPERF_MIN_WORD_LENGTH || u > GPERF_MAX_WORD_LENGTH)
191 29
                return (NULL);
192 150674
        u += http_asso_values[(uint8_t)(e[-1])] +
193 75337
             http_asso_values[(uint8_t)(b[0])];
194 75337
        if (u > GPERF_MAX_HASH_VALUE)
195 1011
                return (NULL);
196 74326
        retval = &http_hdrflg[u];
197 74326
        if (retval->hdr == NULL)
198 277
                return (NULL);
199 74049
        AN(*retval->hdr);
200 74049
        if (!http_hdr_at(*retval->hdr + 1, b, e - b))
201 1035
                return (NULL);
202 73014
        return (retval);
203 75366
}
204
205
/*--------------------------------------------------------------------*/
206
207
static void
208 48828
http_init_hdr(hdr_t hdr, int flg)
209
{
210
        struct http_hdrflg *f;
211
212 48828
        f = http_hdr_flags(hdr->str, hdr->str + hdr->len - 1);
213 48828
        AN(f);
214 48828
        assert(*f->hdr == hdr);
215 48828
        f->flag = flg;
216 48828
}
217
218
void
219 939
HTTP_Init(void)
220
{
221
        struct vsb *vsb;
222
223
#define HTTPH(a, b, c) http_init_hdr(b, c);
224
#include "tbl/http_headers.h"
225
226
        vsb = VSB_new_auto();
227 939
        AN(vsb);
228
        VSB_printf(vsb, "1.1 %s (Varnish/" PACKAGE_BRANCH ")",
229
            heritage.identity);
230 939
        AZ(VSB_finish(vsb));
231 939
        REPLACE(via_hdr, VSB_data(vsb));
232
        VSB_destroy(&vsb);
233
}
234
235
/*--------------------------------------------------------------------
236
 * These two functions are in an incestuous relationship with the
237
 * order of macros in include/tbl/vsl_tags_http.h
238
 *
239
 * The http->logtag is the SLT_*Method enum, and we add to that, to
240
 * get the SLT_ to use.
241
 */
242
243
static void
244 124721
http_VSLH(const struct http *hp, unsigned hdr)
245
{
246
        int i;
247
248 124721
        if (hp->vsl != NULL) {
249 124716
                assert(VXID_TAG(hp->vsl->wid));
250 124716
                i = hdr;
251 124716
                if (i > HTTP_HDR_FIRST)
252 65610
                        i = HTTP_HDR_FIRST;
253 124716
                i += hp->logtag;
254 124716
                VSLbt(hp->vsl, (enum VSL_tag_e)i, hp->hd[hdr]);
255 124716
        }
256 124721
}
257
258
static void
259 1021
http_VSLH_del(const struct http *hp, unsigned hdr)
260
{
261
        int i;
262
263 1021
        if (hp->vsl != NULL) {
264
                /* We don't support unsetting stuff in the first line */
265 1021
                assert (hdr >= HTTP_HDR_FIRST);
266 1021
                assert(VXID_TAG(hp->vsl->wid));
267 1021
                i = (HTTP_HDR_UNSET - HTTP_HDR_METHOD);
268 1021
                i += hp->logtag;
269 1021
                VSLbt(hp->vsl, (enum VSL_tag_e)i, hp->hd[hdr]);
270 1021
        }
271 1021
}
272
273
/*--------------------------------------------------------------------*/
274
275
void
276 5807
http_VSL_log(const struct http *hp)
277
{
278
        unsigned u;
279
280 50721
        for (u = 0; u < hp->nhd; u++)
281 78215
                if (hp->hd[u].b != NULL)
282 33301
                        http_VSLH(hp, u);
283 5807
}
284
285
/*--------------------------------------------------------------------*/
286
287
static void
288 104
http_fail(const struct http *hp)
289
{
290
        char id[WS_ID_SIZE];
291
292 104
        VSC_C_main->losthdr++;
293 104
        WS_Id(hp->ws, id);
294 104
        VSLb(hp->vsl, SLT_Error, "out of workspace (%s)", id);
295 104
        WS_MarkOverflow(hp->ws);
296 104
}
297
298
/*--------------------------------------------------------------------
299
 * List of canonical HTTP response code names from RFC2616
300
 */
301
302
static struct http_msg {
303
        unsigned        nbr;
304
        const char      *status;
305
        const char      *txt;
306
} http_msg[] = {
307
#define HTTP_RESP(n, t) { n, #n, t},
308
#include "tbl/http_response.h"
309
        { 0, "0", NULL }
310
};
311
312
const char *
313 1473
http_Status2Reason(unsigned status, const char **sstr)
314
{
315
        struct http_msg *mp;
316
317 1473
        status %= 1000;
318 1473
        assert(status >= 100);
319 44787
        for (mp = http_msg; mp->nbr != 0 && mp->nbr <= status; mp++)
320 44742
                if (mp->nbr == status) {
321 1428
                        if (sstr)
322 1091
                                *sstr = mp->status;
323 1428
                        return (mp->txt);
324
                }
325 45
        return ("Unknown HTTP Status");
326 1473
}
327
328
/*--------------------------------------------------------------------*/
329
330
unsigned
331 5300
HTTP_estimate(unsigned nhttp)
332
{
333
334
        /* XXX: We trust the structs to size-aligned as necessary */
335 5300
        return (PRNDUP(sizeof(struct http) + sizeof(txt) * nhttp + nhttp));
336
}
337
338
struct http *
339 15900
HTTP_create(void *p, uint16_t nhttp, unsigned len)
340
{
341
        struct http *hp;
342
343 15900
        hp = p;
344 15900
        hp->magic = HTTP_MAGIC;
345 15900
        hp->hd = (void*)(hp + 1);
346 15900
        hp->shd = nhttp;
347 15900
        hp->hdf = (void*)(hp->hd + nhttp);
348 15900
        assert((unsigned char*)p + len == hp->hdf + PRNDUP(nhttp));
349 15900
        return (hp);
350
}
351
352
/*--------------------------------------------------------------------*/
353
354
void
355 14834
HTTP_Setup(struct http *hp, struct ws *ws, struct vsl_log *vsl,
356
    enum VSL_tag_e  whence)
357
{
358 14834
        http_Teardown(hp);
359 14834
        hp->nhd = HTTP_HDR_FIRST;
360 14834
        hp->logtag = whence;
361 14834
        hp->ws = ws;
362 14834
        hp->vsl = vsl;
363 14834
}
364
365
/*--------------------------------------------------------------------
366
 * http_Teardown() is a safety feature, we use it to zap all http
367
 * structs once we're done with them, to minimize the risk that
368
 * old stale pointers exist to no longer valid stuff.
369
 */
370
371
void
372 23160
http_Teardown(struct http *hp)
373
{
374
375 23160
        CHECK_OBJ_NOTNULL(hp, HTTP_MAGIC);
376 23160
        AN(hp->shd);
377 23160
        memset(&hp->nhd, 0, sizeof *hp - offsetof(struct http, nhd));
378 23160
        memset(hp->hd, 0, sizeof *hp->hd * hp->shd);
379 23160
        memset(hp->hdf, 0, sizeof *hp->hdf * hp->shd);
380 23160
}
381
382
/*--------------------------------------------------------------------
383
 * Duplicate the http content into another http
384
 * We cannot just memcpy the struct because the hd & hdf are private
385
 * storage to the struct http.
386
 */
387
388
void
389 6518
HTTP_Dup(struct http *to, const struct http * fm)
390
{
391
392 6518
        assert(fm->nhd <= to->shd);
393 6518
        memcpy(to->hd, fm->hd, fm->nhd * sizeof *to->hd);
394 6518
        memcpy(to->hdf, fm->hdf, fm->nhd * sizeof *to->hdf);
395 6518
        to->nhd = fm->nhd;
396 6518
        to->logtag = fm->logtag;
397 6518
        to->status = fm->status;
398 6518
        to->protover = fm->protover;
399 6518
}
400
401
402
/*--------------------------------------------------------------------
403
 * Clone the entire http structure, including vsl & ws
404
 */
405
406
void
407 6152
HTTP_Clone(struct http *to, const struct http * const fm)
408
{
409
410 6152
        HTTP_Dup(to, fm);
411 6152
        to->vsl = fm->vsl;
412 6152
        to->ws = fm->ws;
413 6152
}
414
415
/*--------------------------------------------------------------------*/
416
417
void
418 6640
http_Proto(struct http *to)
419
{
420
        const char *fm;
421
422 6640
        fm = to->hd[HTTP_HDR_PROTO].b;
423
424 13271
        if (fm != NULL &&
425 6632
            (fm[0] == 'H' || fm[0] == 'h') &&
426 6631
            (fm[1] == 'T' || fm[1] == 't') &&
427 6631
            (fm[2] == 'T' || fm[2] == 't') &&
428 6631
            (fm[3] == 'P' || fm[3] == 'p') &&
429 6631
            fm[4] == '/' &&
430 6631
            vct_isdigit(fm[5]) &&
431 6631
            fm[6] == '.' &&
432 6631
            vct_isdigit(fm[7]) &&
433 6631
            fm[8] == '\0') {
434 6631
                to->protover = 10 * (fm[5] - '0') + (fm[7] - '0');
435 6631
        } else {
436 9
                to->protover = 0;
437
        }
438 6640
}
439
440
/*--------------------------------------------------------------------*/
441
442
void
443 39562
http_SetH(struct http *to, unsigned n, const char *header)
444
{
445
446 39562
        assert(n < to->nhd);
447 39562
        AN(header);
448 39562
        to->hd[n].b = TRUST_ME(header);
449 39562
        to->hd[n].e = strchr(to->hd[n].b, '\0');
450 39562
        to->hdf[n] = 0;
451 39562
        http_VSLH(to, n);
452 39562
        if (n == HTTP_HDR_PROTO)
453 1356
                http_Proto(to);
454 39562
}
455
456
/*--------------------------------------------------------------------*/
457
458
static void
459 36
http_PutField(struct http *to, int field, const char *string)
460
{
461
        const char *p;
462
463 36
        CHECK_OBJ_NOTNULL(to, HTTP_MAGIC);
464 36
        p = WS_Copy(to->ws, string, -1);
465 36
        if (p == NULL) {
466 1
                http_fail(to);
467 1
                VSLbs(to->vsl, SLT_LostHeader, TOSTRAND(string));
468 1
                return;
469
        }
470 35
        http_SetH(to, field, p);
471 36
}
472
473
/*--------------------------------------------------------------------*/
474
475
int
476 144457
http_IsHdr(const txt *hh, hdr_t hdr)
477
{
478
479 144457
        Tcheck(*hh);
480 144457
        CHECK_HDR(hdr);
481 144457
        return (http_hdr_at(hdr->str, hh->b, hdr->len));
482
}
483
484
/*--------------------------------------------------------------------*/
485
486
static unsigned
487 148508
http_findhdr(const struct http *hp, unsigned l, const char *hdr)
488
{
489
        unsigned u;
490
491 641309
        for (u = HTTP_HDR_FIRST; u < hp->nhd; u++) {
492 525799
                Tcheck(hp->hd[u]);
493 525799
                if (hp->hd[u].e < hp->hd[u].b + l + 1)
494 95241
                        continue;
495 430558
                if (hp->hd[u].b[l] != ':')
496 361735
                        continue;
497 68823
                if (!http_hdr_at(hdr, hp->hd[u].b, l))
498 35825
                        continue;
499 32998
                return (u);
500
        }
501 115510
        return (0);
502 148508
}
503
504
/*--------------------------------------------------------------------
505
 * Count how many instances we have of this header
506
 */
507
508
unsigned
509 7544
http_CountHdr(const struct http *hp, hdr_t hdr)
510
{
511 7544
        unsigned retval = 0;
512
        unsigned u;
513
514 7544
        CHECK_OBJ_NOTNULL(hp, HTTP_MAGIC);
515
516 25158
        for (u = HTTP_HDR_FIRST; u < hp->nhd; u++) {
517 17614
                Tcheck(hp->hd[u]);
518 17614
                if (http_IsHdr(&hp->hd[u], hdr))
519 3730
                        retval++;
520 17614
        }
521 7544
        return (retval);
522
}
523
524
/*--------------------------------------------------------------------
525
 * This function collapses multiple header lines of the same name.
526
 * The lines are joined with a comma, according to [rfc2616, 4.2bot, p32]
527
 */
528
529
void
530 24250
http_CollectHdr(struct http *hp, hdr_t hdr)
531
{
532
533 24250
        http_CollectHdrSep(hp, hdr, NULL);
534 24250
}
535
536
/*--------------------------------------------------------------------
537
 * You may prefer to collapse header fields using a different separator.
538
 * For Cookie headers, the separator is "; " for example. That's probably
539
 * the only example too.
540
 */
541
542
void
543 24397
http_CollectHdrSep(struct http *hp, hdr_t hdr, const char *sep)
544
{
545
        unsigned u, lsep, ml, f, x, d;
546 24397
        char *b = NULL, *e = NULL;
547
        const char *v;
548
549 24397
        CHECK_OBJ_NOTNULL(hp, HTTP_MAGIC);
550 24397
        CHECK_HDR(hdr);
551
552 24397
        if (WS_Overflowed(hp->ws))
553 20
                return;
554
555 24377
        if (sep == NULL || *sep == '\0')
556 24231
                sep = ", ";
557 24377
        lsep = strlen(sep);
558
559 24377
        f = http_findhdr(hp, hdr->len - 1, hdr->str);
560 24377
        if (f == 0)
561 23969
                return;
562
563 1503
        for (d = u = f + 1; u < hp->nhd; u++) {
564 1095
                Tcheck(hp->hd[u]);
565 1095
                if (!http_IsHdr(&hp->hd[u], hdr)) {
566 1077
                        if (d != u) {
567 71
                                hp->hd[d] = hp->hd[u];
568 71
                                hp->hdf[d] = hp->hdf[u];
569 71
                        }
570 1077
                        d++;
571 1077
                        continue;
572
                }
573 18
                if (b == NULL) {
574
                        /* Found second header, start our collection */
575 16
                        ml = WS_ReserveAll(hp->ws);
576 16
                        b = WS_Reservation(hp->ws);
577 16
                        e = b + ml;
578 16
                        x = Tlen(hp->hd[f]);
579 16
                        if (b + x >= e) {
580 0
                                http_fail(hp);
581 0
                                VSLbs(hp->vsl, SLT_LostHeader,
582 0
                                    TOSTRAND(hdr->str));
583 0
                                WS_Release(hp->ws, 0);
584 0
                                return;
585
                        }
586 16
                        memcpy(b, hp->hd[f].b, x);
587 16
                        b += x;
588 16
                }
589
590 18
                AN(b);
591 18
                AN(e);
592
593
                /* Append the Nth header we found */
594 18
                x = Tlen(hp->hd[u]) - hdr->len;
595
596 18
                v = hp->hd[u].b + hdr->len;
597 36
                while (vct_issp(*v)) {
598 18
                        v++;
599 18
                        x--;
600
                }
601
602 18
                if (b + lsep + x >= e) {
603 0
                        http_fail(hp);
604 0
                        VSLbs(hp->vsl, SLT_LostHeader, TOSTRAND(hdr->str));
605 0
                        WS_Release(hp->ws, 0);
606 0
                        return;
607
                }
608 18
                memcpy(b, sep, lsep);
609 18
                b += lsep;
610 18
                memcpy(b, v, x);
611 18
                b += x;
612 18
        }
613 408
        if (b == NULL)
614 392
                return;
615 16
        hp->nhd = (uint16_t)d;
616 16
        AN(e);
617 16
        *b = '\0';
618 16
        hp->hd[f].b = WS_Reservation(hp->ws);
619 16
        hp->hd[f].e = b;
620 16
        WS_ReleaseP(hp->ws, b + 1);
621 24397
}
622
623
/*--------------------------------------------------------------------*/
624
625
int
626 123986
http_GetHdr(const struct http *hp, hdr_t hdr, const char **ptr)
627
{
628
        unsigned u;
629
        const char *p;
630
631 123986
        CHECK_HDR(hdr);
632 123986
        u = http_findhdr(hp, hdr->len - 1, hdr->str);
633 123986
        if (u == 0) {
634 91473
                if (ptr != NULL)
635 84024
                        *ptr = NULL;
636 91473
                return (0);
637
        }
638 32513
        if (ptr != NULL) {
639 26618
                p = hp->hd[u].b + hdr->len;
640 53192
                while (vct_issp(*p))
641 26574
                        p++;
642 26618
                *ptr = p;
643 26618
        }
644 32513
        return (1);
645 123986
}
646
647
/*-----------------------------------------------------------------------------
648
 * Split source string at any of the separators, return pointer to first
649
 * and last+1 char of substrings, with whitespace trimmed at both ends.
650
 * If sep being an empty string is shorthand for VCT::SP
651
 * If stop is NULL, src is NUL terminated.
652
 */
653
654
static int
655 2464
http_split(const char **src, const char *stop, const char *sep,
656
    const char **b, const char **e)
657
{
658
        const char *p, *q;
659
660 2464
        AN(src);
661 2464
        AN(*src);
662 2464
        AN(sep);
663 2464
        AN(b);
664 2464
        AN(e);
665
666 2464
        if (stop == NULL)
667 2457
                stop = strchr(*src, '\0');
668
669 4225
        for (p = *src; p < stop && (vct_issp(*p) || strchr(sep, *p)); p++)
670 87
                continue;
671
672 2464
        if (p >= stop) {
673 877
                *b = NULL;
674 877
                *e = NULL;
675 877
                return (0);
676
        }
677
678 1587
        *b = p;
679 1587
        if (*sep == '\0') {
680 0
                for (q = p + 1; q < stop && !vct_issp(*q); q++)
681 0
                        continue;
682 0
                *e = q;
683 0
                *src = q;
684 0
                return (1);
685
        }
686 7691
        for (q = p + 1; q < stop && !strchr(sep, *q); q++)
687 6104
                continue;
688 1587
        *src = q;
689 1587
        while (q > p && vct_issp(q[-1]))
690 0
                q--;
691 1587
        *e = q;
692 1587
        return (1);
693 2464
}
694
695
/*-----------------------------------------------------------------------------
696
 * Comparison rule for tokens:
697
 *      if target string starts with '"', we use memcmp() and expect closing
698
 *      double quote as well
699
 *      otherwise we use http_tok_at()
700
 *
701
 * On match we increment *bp past the token name.
702
 */
703
704
static int
705 1401
http_istoken(const char **bp, const char *e, const char *token)
706
{
707 1401
        int fl = strlen(token);
708
        const char *b;
709
710 1401
        AN(bp);
711 1401
        AN(e);
712 1401
        AN(token);
713
714 1401
        b = *bp;
715
716 1401
        if (b + fl + 2 <= e && *b == '"' &&
717 0
            !memcmp(b + 1, token, fl) && b[fl + 1] == '"') {
718 0
                *bp += fl + 2;
719 0
                return (1);
720
        }
721 1441
        if (b + fl <= e && http_tok_at(b, token, fl) &&
722 660
            (b + fl == e || !vct_istchar(b[fl]))) {
723 660
                *bp += fl;
724 660
                return (1);
725
        }
726 741
        return (0);
727 1401
}
728
729
/*-----------------------------------------------------------------------------
730
 * Find a given data element (token) in a header according to RFC2616's #rule
731
 * (section 2.1, p15)
732
 *
733
 * On case sensitivity:
734
 *
735
 * Section 4.2 (Messages Headers) defines field (header) name as case
736
 * insensitive, but the field (header) value/content may be case-sensitive.
737
 *
738
 * http_GetHdrToken looks up a token in a header value and the rfc does not say
739
 * explicitly if tokens are to be compared with or without respect to case.
740
 *
741
 * But all examples and specific statements regarding tokens follow the rule
742
 * that unquoted tokens are to be matched case-insensitively and quoted tokens
743
 * case-sensitively.
744
 *
745
 * The optional pb and pe arguments will point to the token content start and
746
 * end+1, white space trimmed on both sides.
747
 */
748
749
int
750 14829
http_GetHdrToken(const struct http *hp, hdr_t hdr,
751
    const char *token, const char **pb, const char **pe)
752
{
753
        const char *h, *b, *e;
754
755 14829
        if (pb != NULL)
756 11191
                *pb = NULL;
757 14829
        if (pe != NULL)
758 3512
                *pe = NULL;
759 14829
        if (!http_GetHdr(hp, hdr, &h))
760 13465
                return (0);
761 1364
        AN(h);
762
763 2105
        while (http_split(&h, NULL, ",", &b, &e))
764 1401
                if (http_istoken(&b, e, token))
765 660
                        break;
766 1364
        if (b == NULL)
767 704
                return (0);
768 660
        if (pb != NULL) {
769 656
                for (; vct_islws(*b); b++)
770 5
                        continue;
771 651
                if (b == e) {
772 612
                        b = NULL;
773 612
                        e = NULL;
774 612
                }
775 651
                *pb = b;
776 651
                if (pe != NULL)
777 605
                        *pe = e;
778 651
        }
779 660
        return (1);
780 14829
}
781
782
/*--------------------------------------------------------------------
783
 * Find a given header field's quality value (qvalue).
784
 */
785
786
double
787 3512
http_GetHdrQ(const struct http *hp, hdr_t hdr, const char *field)
788
{
789
        const char *hb, *he, *b, *e;
790
        int i;
791
        double a, f;
792
793 3512
        i = http_GetHdrToken(hp, hdr, field, &hb, &he);
794 3512
        if (!i)
795 2907
                return (0.);
796
797 605
        if (hb == NULL)
798 598
                return (1.);
799 7
        while (http_split(&hb, he, ";", &b, &e)) {
800 7
                if (*b != 'q')
801 0
                        continue;
802 7
                for (b++; b < e && vct_issp(*b); b++)
803 0
                        continue;
804 7
                if (b == e || *b != '=')
805 0
                        continue;
806 7
                break;
807
        }
808 7
        if (b == NULL)
809 0
                return (1.);
810 7
        for (b++; b < e && vct_issp(*b); b++)
811 0
                continue;
812 7
        if (b == e || (*b != '.' && !vct_isdigit(*b)))
813 1
                return (0.);
814 6
        a = 0;
815 12
        while (b < e && vct_isdigit(*b)) {
816 6
                a *= 10.;
817 6
                a += *b - '0';
818 6
                b++;
819
        }
820 6
        if (b == e || *b++ != '.')
821 1
                return (a);
822 5
        f = .1;
823 16
        while (b < e && vct_isdigit(*b)) {
824 11
                a += f * (*b - '0');
825 11
                f *= .1;
826 11
                b++;
827
        }
828 5
        return (a);
829 3512
}
830
831
/*--------------------------------------------------------------------
832
 * Find a given header field's value.
833
 */
834
835
int
836 7681
http_GetHdrField(const struct http *hp, hdr_t hdr,
837
    const char *field, const char **ptr)
838
{
839
        const char *h;
840
        int i;
841
842 7681
        if (ptr != NULL)
843 3899
                *ptr = NULL;
844
845 7681
        h = NULL;
846 7681
        i = http_GetHdrToken(hp, hdr, field, &h, NULL);
847 7681
        if (!i)
848 7635
                return (i);
849
850 46
        if (ptr != NULL && h != NULL) {
851
                /* Skip whitespace, looking for '=' */
852 32
                while (*h && vct_issp(*h))
853 0
                        h++;
854 32
                if (*h == '=') {
855 32
                        h++;
856 36
                        while (*h && vct_issp(*h))
857 4
                                h++;
858 32
                        *ptr = h;
859 32
                }
860 32
        }
861 46
        return (i);
862 7681
}
863
864
/*--------------------------------------------------------------------*/
865
866
ssize_t
867 11102
http_GetContentLength(const struct http *hp)
868
{
869
        ssize_t cl;
870
        const char *b;
871
872 11102
        CHECK_OBJ_NOTNULL(hp, HTTP_MAGIC);
873
874 11102
        if (!http_GetHdr(hp, H_Content_Length, &b))
875 4363
                return (-1);
876 6739
        cl = VNUM_uint(b, NULL, &b);
877 6739
        if (cl < 0)
878 4
                return (-2);
879 6735
        while (vct_islws(*b))
880 0
                b++;
881 6735
        if (*b != '\0')
882 0
                return (-2);
883 6735
        return (cl);
884 11102
}
885
886
ssize_t
887 1988
http_GetContentRange(const struct http *hp, ssize_t *lo, ssize_t *hi)
888
{
889
        ssize_t tmp, cl;
890
        const char *b, *t;
891
892 1988
        CHECK_OBJ_NOTNULL(hp, HTTP_MAGIC);
893
894 1988
        if (lo == NULL)
895 0
                lo = &tmp;
896 1988
        if (hi == NULL)
897 0
                hi = &tmp;
898
899 1988
        *lo = *hi = -1;
900
901 1988
        if (!http_GetHdr(hp, H_Content_Range, &b))
902 1979
                return (-1);
903
904 9
        t = strchr(b, ' ');
905 9
        if (t == NULL)
906 1
                return (-2);            // Missing space after range unit
907
908 8
        if (!http_range_at(b, bytes, t - b))
909 1
                return (-1);            // Unknown range unit, ignore
910 7
        b = t + 1;
911
912 7
        if (*b == '*') {                // Content-Range: bytes */123
913 3
                *lo = *hi = -1;
914 3
                b++;
915 3
        } else {                        // Content-Range: bytes 1-2/3
916 4
                *lo = VNUM_uint(b, NULL, &b);
917 4
                if (*lo < 0)
918 0
                        return (-2);
919 4
                if (*b != '-')
920 0
                        return (-2);
921 4
                *hi = VNUM_uint(b + 1, NULL, &b);
922 4
                if (*hi < 0)
923 0
                        return (-2);
924
        }
925 7
        if (*b != '/')
926 0
                return (-2);
927 7
        if (b[1] == '*') {              // Content-Range: bytes 1-2/*
928 1
                cl = -1;
929 1
                b += 2;
930 1
        } else {
931 6
                cl = VNUM_uint(b + 1, NULL, &b);
932 6
                if (cl <= 0)
933 0
                        return (-2);
934
        }
935 7
        while (vct_islws(*b))
936 0
                b++;
937 7
        if (*b != '\0')
938 0
                return (-2);
939 7
        if (*lo > *hi)
940 0
                return (-2);
941 7
        assert(cl >= -1);
942 7
        if (*lo >= cl || *hi >= cl)
943 1
                return (-2);
944 6
        AN(cl);
945 6
        return (cl);
946 1988
}
947
948
const char *
949 2037
http_GetRange(const struct http *hp, ssize_t *lo, ssize_t *hi, ssize_t len)
950
{
951
        ssize_t tmp_lo, tmp_hi;
952
        const char *b, *t;
953
954 2037
        CHECK_OBJ_NOTNULL(hp, HTTP_MAGIC);
955
956 2037
        if (lo == NULL)
957 0
                lo = &tmp_lo;
958 2037
        if (hi == NULL)
959 0
                hi = &tmp_hi;
960
961 2037
        *lo = *hi = -1;
962
963 2037
        if (!http_GetHdr(hp, H_Range, &b))
964 1983
                return (NULL);
965
966 54
        t = strchr(b, '=');
967 54
        if (t == NULL)
968 2
                return ("Missing '='");
969
970 52
        if (!http_range_at(b, bytes, t - b))
971 3
                return ("Not Bytes");
972 49
        b = t + 1;
973
974 49
        *lo = VNUM_uint(b, NULL, &b);
975 49
        if (*lo == -2)
976 0
                return ("Low number too big");
977 49
        if (*b++ != '-')
978 0
                return ("Missing hyphen");
979
980 49
        *hi = VNUM_uint(b, NULL, &b);
981 49
        if (*hi == -2)
982 0
                return ("High number too big");
983 49
        if (*lo == -1 && *hi == -1)
984 1
                return ("Neither high nor low");
985 48
        if (*lo == -1 && *hi == 0)
986 1
                return ("No low, high is zero");
987 47
        if (*hi >= 0 && *hi < *lo)
988 1
                return ("high smaller than low");
989
990 47
        while (vct_islws(*b))
991 1
                b++;
992 46
        if (*b != '\0')
993 1
                return ("Trailing stuff");
994
995 45
        assert(*lo >= -1);
996 45
        assert(*hi >= -1);
997
998 45
        if (*lo < 0) {
999 4
                assert(*hi > 0);
1000 4
                *lo = len - *hi;
1001 4
                if (*lo < 0)
1002 2
                        *lo = 0;
1003 4
                *hi = len - 1;
1004 45
        } else if (len >= 0 && (*hi >= len || *hi < 0)) {
1005 9
                *hi = len - 1;
1006 9
        }
1007
1008 45
        if (len <= 0)
1009 17
                return (NULL);                  // Allow 200 response
1010
1011 28
        if (*lo >= len)
1012 1
                return ("low range beyond object");
1013
1014 27
        return (NULL);
1015 2037
}
1016
1017
/*--------------------------------------------------------------------
1018
 */
1019
1020
stream_close_t
1021 5837
http_DoConnection(struct http *hp, stream_close_t sc_close)
1022
{
1023
        const char *h, *b, *e;
1024
        stream_close_t retval;
1025
        unsigned u, v;
1026
        struct http_hdrflg *f;
1027
1028 5837
        CHECK_OBJ_NOTNULL(hp, HTTP_MAGIC);
1029 5837
        assert(sc_close == SC_REQ_CLOSE || sc_close == SC_RESP_CLOSE);
1030
1031 5837
        if (hp->protover == 10)
1032 38
                retval = SC_REQ_HTTP10;
1033
        else
1034 5799
                retval = SC_NULL;
1035
1036 5837
        http_CollectHdr(hp, H_Connection);
1037 5837
        if (!http_GetHdr(hp, H_Connection, &h))
1038 5661
                return (retval);
1039 176
        AN(h);
1040 352
        while (http_split(&h, NULL, ",", &b, &e)) {
1041 179
                u = pdiff(b, e);
1042 179
                if (u == 5 && http_hdr_at(b, "close", u))
1043 144
                        retval = sc_close;
1044 179
                if (u == 10 && http_hdr_at(b, "keep-alive", u))
1045 23
                        retval = SC_NULL;
1046
1047
                /* Refuse removal of well-known-headers if they would pass. */
1048 179
                f = http_hdr_flags(b, e);
1049 179
                if (f != NULL && !(f->flag & HTTPH_R_PASS))
1050 3
                        return (SC_RX_BAD);
1051
1052 856
                for (v = HTTP_HDR_FIRST; v < hp->nhd; v++) {
1053 680
                        Tcheck(hp->hd[v]);
1054 680
                        if (hp->hd[v].e < hp->hd[v].b + u + 1)
1055 31
                                continue;
1056 649
                        if (hp->hd[v].b[u] != ':')
1057 599
                                continue;
1058 50
                        if (!http_hdr_at(b, hp->hd[v].b, u))
1059 38
                                continue;
1060 12
                        hp->hdf[v] |= HDF_FILTER;
1061 12
                }
1062
        }
1063 173
        CHECK_OBJ_NOTNULL(retval, STREAM_CLOSE_MAGIC);
1064 173
        return (retval);
1065 5837
}
1066
1067
/*--------------------------------------------------------------------*/
1068
1069
int
1070 11107
http_HdrIs(const struct http *hp, hdr_t hdr, const char *val)
1071
{
1072
        const char *p;
1073
1074 11107
        if (!http_GetHdr(hp, hdr, &p))
1075 10514
                return (0);
1076 593
        AN(p);
1077 593
        if (http_tok_eq(p, val))
1078 567
                return (1);
1079 26
        return (0);
1080 11107
}
1081
1082
/*--------------------------------------------------------------------*/
1083
1084
uint16_t
1085 14364
http_GetStatus(const struct http *hp)
1086
{
1087
1088 14364
        CHECK_OBJ_NOTNULL(hp, HTTP_MAGIC);
1089 14364
        return (hp->status);
1090
}
1091
1092
int
1093 9972
http_IsStatus(const struct http *hp, int val)
1094
{
1095
1096 9972
        CHECK_OBJ_NOTNULL(hp, HTTP_MAGIC);
1097 9972
        assert(val >= 100 && val <= 999);
1098 9972
        return (val == (hp->status % 1000));
1099
}
1100
1101
/*--------------------------------------------------------------------
1102
 * Setting the status will also set the Reason appropriately
1103
 */
1104
1105
void
1106 1127
http_SetStatus(struct http *to, uint16_t status, const char *reason)
1107
{
1108
        char buf[4];
1109 1127
        const char *sstr = NULL;
1110
1111 1127
        CHECK_OBJ_NOTNULL(to, HTTP_MAGIC);
1112
        /*
1113
         * We allow people to use top digits for internal VCL
1114
         * signalling, but strip them from the ASCII version.
1115
         */
1116 1127
        to->status = status;
1117 1127
        status %= 1000;
1118 1127
        assert(status >= 100);
1119
1120 1127
        if (reason == NULL)
1121 231
                reason = http_Status2Reason(status, &sstr);
1122
        else
1123 896
                (void)http_Status2Reason(status, &sstr);
1124
1125 1127
        if (sstr) {
1126 1091
                http_SetH(to, HTTP_HDR_STATUS, sstr);
1127 1091
        } else {
1128 36
                bprintf(buf, "%03d", status);
1129 36
                http_PutField(to, HTTP_HDR_STATUS, buf);
1130
        }
1131 1127
        http_SetH(to, HTTP_HDR_REASON, reason);
1132 1127
}
1133
1134
/*--------------------------------------------------------------------*/
1135
1136
const char *
1137 5241
http_GetMethod(const struct http *hp)
1138
{
1139
1140 5241
        CHECK_OBJ_NOTNULL(hp, HTTP_MAGIC);
1141 5241
        Tcheck(hp->hd[HTTP_HDR_METHOD]);
1142 5241
        return (hp->hd[HTTP_HDR_METHOD].b);
1143
}
1144
1145
/*--------------------------------------------------------------------
1146
 * Force a particular header field to a particular value
1147
 */
1148
1149
void
1150 7707
http_ForceField(struct http *to, unsigned n, const char *t)
1151
{
1152
        int i;
1153
1154 7707
        CHECK_OBJ_NOTNULL(to, HTTP_MAGIC);
1155 7707
        assert(n < HTTP_HDR_FIRST);
1156 7707
        assert(n == HTTP_HDR_METHOD || n == HTTP_HDR_PROTO);
1157 7707
        AN(t);
1158
1159
        /* NB: method names and protocol versions are case-sensitive. */
1160 7707
        if (to->hd[n].b == NULL || strcmp(to->hd[n].b, t)) {
1161 119
                i = (HTTP_HDR_UNSET - HTTP_HDR_METHOD);
1162 119
                i += to->logtag;
1163
                /* XXX: this is a dead branch */
1164 119
                if (n >= HTTP_HDR_FIRST)
1165 0
                        VSLbt(to->vsl, (enum VSL_tag_e)i, to->hd[n]);
1166 119
                http_SetH(to, n, t);
1167 119
        }
1168 7707
}
1169
1170
/*--------------------------------------------------------------------*/
1171
1172
void
1173 1029
http_PutResponse(struct http *to, const char *proto, uint16_t status,
1174
    const char *reason)
1175
{
1176
1177 1029
        CHECK_OBJ_NOTNULL(to, HTTP_MAGIC);
1178 1029
        if (proto != NULL)
1179 1029
                http_SetH(to, HTTP_HDR_PROTO, proto);
1180 1029
        http_SetStatus(to, status, reason);
1181 1029
}
1182
1183
/*--------------------------------------------------------------------
1184
 * check if header is filtered by the dynamic marker or the static
1185
 * definitions in http_headers.h
1186
 */
1187
1188
static inline int
1189 39700
http_isfiltered(const struct http *fm, unsigned u, unsigned how)
1190
{
1191
        const char *e;
1192
        const struct http_hdrflg *f;
1193
1194 39700
        if (fm->hdf[u] & HDF_FILTER)
1195 17
                return (1);
1196 39683
        if (u < HTTP_HDR_FIRST)
1197 13323
                return (0);
1198 26360
        e = strchr(fm->hd[u].b, ':');
1199 26360
        if (e == NULL)
1200 0
                return (0);
1201 26360
        f = http_hdr_flags(fm->hd[u].b, e);
1202 26360
        return (f != NULL && f->flag & how);
1203 39700
}
1204
1205
int
1206 840
http_IsFiltered(const struct http *fm, unsigned u, unsigned how)
1207
{
1208
1209 840
        return (http_isfiltered(fm, u, how));
1210
}
1211
1212
/*--------------------------------------------------------------------
1213
 * Estimate how much workspace we need to Filter this header according
1214
 * to 'how'.
1215
 */
1216
1217
unsigned
1218 2224
http_EstimateWS(const struct http *fm, unsigned how)
1219
{
1220
        unsigned u, l;
1221
1222 2224
        l = 4;
1223 2224
        CHECK_OBJ_NOTNULL(fm, HTTP_MAGIC);
1224 21239
        for (u = 0; u < fm->nhd; u++) {
1225 19015
                if (u == HTTP_HDR_METHOD || u == HTTP_HDR_URL)
1226 4448
                        continue;
1227 14567
                Tcheck(fm->hd[u]);
1228 14567
                if (http_isfiltered(fm, u, how))
1229 260
                        continue;
1230 14307
                l += Tlen(fm->hd[u]) + 1L;
1231 14307
        }
1232 2224
        return (PRNDUP(l + 1L));
1233
}
1234
1235
/*--------------------------------------------------------------------
1236
 * Encode http struct as byte string.
1237
 *
1238
 * XXX: We could save considerable special-casing below by encoding also
1239
 * XXX: H__Status, H__Reason and H__Proto into the string, but it would
1240
 * XXX: add 26-30 bytes to all encoded objects to save a little code.
1241
 * XXX: It could possibly be a good idea for later HTTP versions.
1242
 */
1243
1244
void
1245 2218
HTTP_Encode(const struct http *fm, uint8_t *p0, unsigned l, unsigned how)
1246
{
1247
        unsigned u, w;
1248
        uint16_t n;
1249
        uint8_t *p, *e;
1250
1251 2218
        AN(p0);
1252 2218
        AN(l);
1253 2218
        p = p0;
1254 2218
        e = p + l;
1255 2218
        assert(p + 5 <= e);
1256 2218
        assert(fm->nhd <= fm->shd);
1257 2218
        n = HTTP_HDR_FIRST - 3;
1258 2218
        vbe16enc(p + 2, fm->status);
1259 2218
        p += 4;
1260 2218
        CHECK_OBJ_NOTNULL(fm, HTTP_MAGIC);
1261 21176
        for (u = 0; u < fm->nhd; u++) {
1262 18958
                if (u == HTTP_HDR_METHOD || u == HTTP_HDR_URL)
1263 4436
                        continue;
1264 14522
                Tcheck(fm->hd[u]);
1265 14522
                if (http_isfiltered(fm, u, how))
1266 260
                        continue;
1267 14262
                http_VSLH(fm, u);
1268 14262
                w = Tlen(fm->hd[u]) + 1L;
1269 14262
                assert(p + w + 1 <= e);
1270 14262
                memcpy(p, fm->hd[u].b, w);
1271 14262
                p += w;
1272 14262
                n++;
1273 14262
        }
1274 2218
        *p++ = '\0';
1275 2218
        assert(p <= e);
1276 2218
        vbe16enc(p0, n + 1);
1277 2218
}
1278
1279
/*--------------------------------------------------------------------
1280
 * Decode byte string into http struct
1281
 */
1282
1283
int
1284 3268
HTTP_Decode(struct http *to, const uint8_t *fm)
1285
{
1286
1287 3268
        CHECK_OBJ_NOTNULL(to, HTTP_MAGIC);
1288 3268
        AN(to->vsl);
1289 3268
        AN(fm);
1290 3268
        if (vbe16dec(fm) <= to->shd) {
1291 3266
                to->status = vbe16dec(fm + 2);
1292 3266
                fm += 4;
1293 30927
                for (to->nhd = 0; to->nhd < to->shd; to->nhd++) {
1294 30927
                        if (to->nhd == HTTP_HDR_METHOD ||
1295 27654
                            to->nhd == HTTP_HDR_URL) {
1296 6532
                                to->hd[to->nhd].b = NULL;
1297 6532
                                to->hd[to->nhd].e = NULL;
1298 6532
                                continue;
1299
                        }
1300 24395
                        if (*fm == '\0')
1301 3266
                                return (0);
1302 21129
                        to->hd[to->nhd].b = (const void*)fm;
1303 21129
                        fm = (const void*)strchr((const void*)fm, '\0');
1304 21129
                        to->hd[to->nhd].e = (const void*)fm;
1305 21129
                        fm++;
1306 21129
                        http_VSLH(to, to->nhd);
1307 21129
                }
1308 0
        }
1309 4
        VSLb(to->vsl, SLT_Error,
1310
            "Too many headers to Decode object (%u vs. %u)",
1311 2
            vbe16dec(fm), to->shd);
1312 2
        return (-1);
1313 3268
}
1314
1315
/*--------------------------------------------------------------------*/
1316
1317
uint16_t
1318 1
HTTP_GetStatusPack(struct worker *wrk, struct objcore *oc)
1319
{
1320
        const char *ptr;
1321 1
        ptr = ObjGetAttr(wrk, oc, OA_HEADERS, NULL);
1322 1
        AN(ptr);
1323
1324 1
        return (vbe16dec(ptr + 2));
1325
}
1326
1327
/*--------------------------------------------------------------------*/
1328
1329
/* Get the first packed header */
1330
int
1331 748
HTTP_IterHdrPack(struct worker *wrk, struct objcore *oc, const char **p)
1332
{
1333
        const char *ptr;
1334
1335 748
        CHECK_OBJ_NOTNULL(wrk, WORKER_MAGIC);
1336 748
        CHECK_OBJ_NOTNULL(oc, OBJCORE_MAGIC);
1337 748
        AN(p);
1338
1339 748
        if (*p == NULL) {
1340 236
                ptr = ObjGetAttr(wrk, oc, OA_HEADERS, NULL);
1341 236
                AN(ptr);
1342 236
                ptr += 4;       /* Skip nhd and status */
1343 236
                ptr = strchr(ptr, '\0') + 1;    /* Skip :proto: */
1344 236
                ptr = strchr(ptr, '\0') + 1;    /* Skip :status: */
1345 236
                ptr = strchr(ptr, '\0') + 1;    /* Skip :reason: */
1346 236
                *p = ptr;
1347 236
        } else {
1348 512
                *p = strchr(*p, '\0') + 1;      /* Skip to next header */
1349
        }
1350 748
        if (**p == '\0')
1351 44
                return (0);
1352 704
        return (1);
1353 748
}
1354
1355
const char *
1356 252
HTTP_GetHdrPack(struct worker *wrk, struct objcore *oc, hdr_t hdr)
1357
{
1358
        const char *ptr;
1359
1360 252
        CHECK_OBJ_NOTNULL(wrk, WORKER_MAGIC);
1361 252
        CHECK_OBJ_NOTNULL(oc, OBJCORE_MAGIC);
1362 252
        CHECK_HDR(hdr);
1363
1364 252
        if (hdr->str[0] == ':') {
1365
                /* Special cases */
1366 16
                ptr = ObjGetAttr(wrk, oc, OA_HEADERS, NULL);
1367 16
                AN(ptr);
1368 16
                ptr += 4;       /* Skip nhd and status */
1369
1370
                /* XXX: should we also have h2_hdr_eq() ? */
1371 16
                if (!strcmp(hdr->str, ":proto:"))
1372 1
                        return (ptr);
1373 15
                ptr = strchr(ptr, '\0') + 1;
1374 15
                if (!strcmp(hdr->str, ":status:"))
1375 14
                        return (ptr);
1376 1
                ptr = strchr(ptr, '\0') + 1;
1377 1
                if (!strcmp(hdr->str, ":reason:"))
1378 1
                        return (ptr);
1379 0
                WRONG("Unknown magic packed header");
1380 0
        }
1381
1382 748
        HTTP_FOREACH_PACK(wrk, oc, ptr) {
1383 704
                if (http_hdr_at(ptr, hdr->str, hdr->len)) {
1384 192
                        ptr += hdr->len;
1385 383
                        while (vct_islws(*ptr))
1386 191
                                ptr++;
1387 192
                        return (ptr);
1388
                }
1389
        }
1390
1391 44
        return (NULL);
1392 252
}
1393
1394
/*--------------------------------------------------------------------
1395
 * Merge any headers in the oc->OA_HEADER into the struct http if they
1396
 * are not there already.
1397
 */
1398
1399
void
1400 28
HTTP_Merge(struct worker *wrk, struct objcore *oc, struct http *to)
1401
{
1402
        const char *ptr;
1403
        unsigned u;
1404
        const char *p;
1405
        unsigned nhd_before_merge;
1406
1407 28
        CHECK_OBJ_NOTNULL(wrk, WORKER_MAGIC);
1408 28
        CHECK_OBJ_NOTNULL(oc, OBJCORE_MAGIC);
1409 28
        CHECK_OBJ_NOTNULL(to, HTTP_MAGIC);
1410
1411 28
        ptr = ObjGetAttr(wrk, oc, OA_HEADERS, NULL);
1412 28
        AN(ptr);
1413
1414 28
        to->status = vbe16dec(ptr + 2);
1415 28
        ptr += 4;
1416
1417 168
        for (u = 0; u < HTTP_HDR_FIRST; u++) {
1418 140
                if (u == HTTP_HDR_METHOD || u == HTTP_HDR_URL)
1419 56
                        continue;
1420 84
                http_SetH(to, u, ptr);
1421 84
                ptr = strchr(ptr, '\0') + 1;
1422 84
        }
1423 28
        nhd_before_merge = to->nhd;
1424 165
        while (*ptr != '\0') {
1425 137
                p = strchr(ptr, ':');
1426 137
                AN(p);
1427 137
                u = http_findhdr(to, p - ptr, ptr);
1428 137
                if (u == 0 || u >= nhd_before_merge)
1429 61
                        http_SetHeader(to, ptr);
1430 137
                ptr = strchr(ptr, '\0') + 1;
1431
        }
1432 28
}
1433
1434
/*--------------------------------------------------------------------*/
1435
1436
static void
1437 6834
http_linkh(const struct http *to, const struct http *fm, unsigned n)
1438
{
1439
1440 6834
        assert(n < HTTP_HDR_FIRST);
1441 6834
        Tcheck(fm->hd[n]);
1442 6834
        to->hd[n] = fm->hd[n];
1443 6834
        to->hdf[n] = fm->hdf[n];
1444 6834
        http_VSLH(to, n);
1445 6834
}
1446
1447
/*--------------------------------------------------------------------*/
1448
1449
void
1450 2278
http_FilterReq(struct http *to, const struct http *fm, unsigned how)
1451
{
1452
        unsigned u;
1453
1454 2278
        CHECK_OBJ_NOTNULL(to, HTTP_MAGIC);
1455 2278
        CHECK_OBJ_NOTNULL(fm, HTTP_MAGIC);
1456
1457 2278
        http_linkh(to, fm, HTTP_HDR_METHOD);
1458 2278
        http_linkh(to, fm, HTTP_HDR_URL);
1459 2278
        http_linkh(to, fm, HTTP_HDR_PROTO);
1460 2278
        to->protover = fm->protover;
1461 2278
        to->status = fm->status;
1462
1463 2278
        to->nhd = HTTP_HDR_FIRST;
1464 12053
        for (u = HTTP_HDR_FIRST; u < fm->nhd; u++) {
1465 9775
                Tcheck(fm->hd[u]);
1466 9775
                if (http_isfiltered(fm, u, how))
1467 115
                        continue;
1468 9660
                assert (to->nhd < to->shd);
1469 9660
                to->hd[to->nhd] = fm->hd[u];
1470 9660
                to->hdf[to->nhd] = 0;
1471 9660
                http_VSLH(to, to->nhd);
1472 9660
                to->nhd++;
1473 9660
        }
1474 2278
}
1475
1476
/*--------------------------------------------------------------------
1477
 * This function copies any header fields which reference foreign
1478
 * storage into our own WS.
1479
 */
1480
1481
void
1482 2247
http_CopyHome(const struct http *hp)
1483
{
1484
        unsigned u, l;
1485
        const char *p;
1486
1487 24413
        for (u = 0; u < hp->nhd; u++) {
1488 22166
                if (hp->hd[u].b == NULL) {
1489 4494
                        assert(u < HTTP_HDR_FIRST);
1490 4494
                        continue;
1491
                }
1492
1493 17672
                l = Tlen(hp->hd[u]);
1494 17672
                if (WS_Allocated(hp->ws, hp->hd[u].b, l))
1495 1409
                        continue;
1496
1497 16263
                p = WS_Copy(hp->ws, hp->hd[u].b, l + 1L);
1498 16263
                if (p == NULL) {
1499 0
                        http_fail(hp);
1500 0
                        VSLbs(hp->vsl, SLT_LostHeader, TOSTRAND(hp->hd[u].b));
1501 0
                        return;
1502
                }
1503 16263
                hp->hd[u].b = p;
1504 16263
                hp->hd[u].e = p + l;
1505 16263
        }
1506 2247
}
1507
1508
/*--------------------------------------------------------------------*/
1509
1510
void
1511 9374
http_SetHeader(struct http *to, const char *header)
1512
{
1513
1514 9374
        CHECK_OBJ_NOTNULL(to, HTTP_MAGIC);
1515 9374
        if (to->nhd >= to->shd) {
1516 1
                VSLbs(to->vsl, SLT_LostHeader, TOSTRAND(header));
1517 1
                http_fail(to);
1518 1
                return;
1519
        }
1520 9373
        http_SetH(to, to->nhd++, header);
1521 9374
}
1522
1523
/*--------------------------------------------------------------------*/
1524
1525
void
1526 4185
http_ForceHeader(struct http *to, hdr_t hdr, const char *val)
1527
{
1528
1529 4185
        CHECK_OBJ_NOTNULL(to, HTTP_MAGIC);
1530 4185
        if (http_HdrIs(to, hdr, val))
1531 387
                return;
1532 3798
        http_Unset(to, hdr);
1533 3798
        http_PrintfHeader(to, "%s %s", hdr->str, val);
1534 4185
}
1535
1536
void
1537 10705
http_AppendHeader(struct http *to, hdr_t hdr, const char *val)
1538
{
1539
        const char *old;
1540
1541 10705
        http_CollectHdr(to, hdr);
1542 10705
        if (http_GetHdr(to, hdr, &old)) {
1543 48
                http_Unset(to, hdr);
1544 48
                http_PrintfHeader(to, "%s %s, %s", hdr->str, old, val);
1545 48
        } else {
1546 10657
                http_PrintfHeader(to, "%s %s", hdr->str, val);
1547
        }
1548 10705
}
1549
1550
void
1551 25162
http_PrintfHeader(struct http *to, const char *fmt, ...)
1552
{
1553
        va_list ap, ap2;
1554
        struct vsb vsb[1];
1555
        size_t sz;
1556
        char *p;
1557
1558 25162
        CHECK_OBJ_NOTNULL(to, HTTP_MAGIC);
1559
1560 25162
        va_start(ap, fmt);
1561 25162
        va_copy(ap2, ap);
1562
1563 25162
        WS_VSB_new(vsb, to->ws);
1564 25162
        VSB_vprintf(vsb, fmt, ap);
1565 25162
        p = WS_VSB_finish(vsb, to->ws, &sz);
1566
1567 25162
        if (p == NULL || to->nhd >= to->shd) {
1568 25
                http_fail(to);
1569 25
                VSLbv(to->vsl, SLT_LostHeader, fmt, ap2);
1570 25
        } else {
1571 25137
                http_SetH(to, to->nhd++, p);
1572
        }
1573 25162
        va_end(ap);
1574 25162
        va_end(ap2);
1575 25162
}
1576
1577
void
1578 1037
http_TimeHeader(struct http *to, const char *fmt, vtim_real now)
1579
{
1580
        char *p;
1581
1582 1037
        CHECK_OBJ_NOTNULL(to, HTTP_MAGIC);
1583 1037
        if (to->nhd >= to->shd) {
1584 0
                VSLbs(to->vsl, SLT_LostHeader, TOSTRAND(fmt));
1585 0
                http_fail(to);
1586 0
                return;
1587
        }
1588 1037
        p = WS_Alloc(to->ws, strlen(fmt) + VTIM_FORMAT_SIZE);
1589 1037
        if (p == NULL) {
1590 77
                http_fail(to);
1591 77
                VSLbs(to->vsl, SLT_LostHeader, TOSTRAND(fmt));
1592 77
                return;
1593
        }
1594 960
        strcpy(p, fmt);
1595 960
        VTIM_format(now, strchr(p, '\0'));
1596 960
        http_SetH(to, to->nhd++, p);
1597 1037
}
1598
1599
const char *
1600 6985
http_ViaHeader(void)
1601
{
1602
1603 6985
        return (via_hdr);
1604
}
1605
1606
/*--------------------------------------------------------------------*/
1607
1608
void
1609 22891
http_Unset(struct http *hp, hdr_t hdr)
1610
{
1611
        uint16_t u, v;
1612
1613 148634
        for (v = u = HTTP_HDR_FIRST; u < hp->nhd; u++) {
1614 125743
                Tcheck(hp->hd[u]);
1615 125743
                if (http_IsHdr(&hp->hd[u], hdr)) {
1616 1021
                        http_VSLH_del(hp, u);
1617 1021
                        continue;
1618
                }
1619 124722
                if (v != u) {
1620 2926
                        memcpy(&hp->hd[v], &hp->hd[u], sizeof *hp->hd);
1621 2926
                        memcpy(&hp->hdf[v], &hp->hdf[u], sizeof *hp->hdf);
1622 2926
                }
1623 124722
                v++;
1624 124722
        }
1625 22891
        hp->nhd = v;
1626 22891
}