Line data Source code
1 : //
2 : // Copyright (c) 2019 Vinnie Falco (vinnie.falco@gmail.com)
3 : // Copyright (c) 2024 Mohammad Nejati
4 : //
5 : // Distributed under the Boost Software License, Version 1.0. (See accompanying
6 : // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
7 : //
8 : // Official repository: https://github.com/cppalliance/http_proto
9 : //
10 :
11 : #include <boost/http_proto/detail/header.hpp>
12 : #include <boost/http_proto/detail/align_up.hpp>
13 : #include <boost/http_proto/field.hpp>
14 : #include <boost/http_proto/fields_view_base.hpp>
15 : #include <boost/http_proto/header_limits.hpp>
16 : #include <boost/http_proto/rfc/list_rule.hpp>
17 : #include <boost/http_proto/rfc/token_rule.hpp>
18 : #include <boost/http_proto/rfc/upgrade_rule.hpp>
19 : #include <boost/http_proto/rfc/detail/rules.hpp>
20 : #include <boost/url/grammar/ci_string.hpp>
21 : #include <boost/url/grammar/parse.hpp>
22 : #include <boost/url/grammar/range_rule.hpp>
23 : #include <boost/url/grammar/recycled.hpp>
24 : #include <boost/url/grammar/unsigned_rule.hpp>
25 : #include <boost/assert.hpp>
26 : #include <boost/assert/source_location.hpp>
27 : #include <boost/static_assert.hpp>
28 : #include <string>
29 : #include <utility>
30 :
31 : #include "../rfc/transfer_encoding_rule.hpp"
32 :
33 : namespace boost {
34 : namespace http_proto {
35 : namespace detail {
36 :
37 : //------------------------------------------------
38 :
39 : auto
40 115 : header::
41 : entry::
42 : operator+(
43 : std::size_t dv) const noexcept ->
44 : entry
45 : {
46 : return {
47 : static_cast<
48 115 : offset_type>(np + dv),
49 115 : nn,
50 : static_cast<
51 115 : offset_type>(vp + dv),
52 115 : vn,
53 115 : id };
54 : }
55 :
56 : auto
57 79 : header::
58 : entry::
59 : operator-(
60 : std::size_t dv) const noexcept ->
61 : entry
62 : {
63 : return {
64 : static_cast<
65 79 : offset_type>(np - dv),
66 79 : nn,
67 : static_cast<
68 79 : offset_type>(vp - dv),
69 79 : vn,
70 79 : id };
71 : }
72 :
73 : //------------------------------------------------
74 :
75 : constexpr
76 : header::
77 : header(fields_tag) noexcept
78 : : kind(detail::kind::fields)
79 : , cbuf("\r\n")
80 : , size(2)
81 : , fld{}
82 : {
83 : }
84 :
85 : constexpr
86 : header::
87 : header(request_tag) noexcept
88 : : kind(detail::kind::request)
89 : , cbuf("GET / HTTP/1.1\r\n\r\n")
90 : , size(18)
91 : , prefix(16)
92 : , req{ 3, 1,
93 : http_proto::method::get }
94 : {
95 : }
96 :
97 : constexpr
98 : header::
99 : header(response_tag) noexcept
100 : : kind(detail::kind::response)
101 : , cbuf("HTTP/1.1 200 OK\r\n\r\n")
102 : , size(19)
103 : , prefix(17)
104 : , res{ 200,
105 : http_proto::status::ok }
106 : {
107 : }
108 :
109 : //------------------------------------------------
110 :
111 : header const*
112 249 : header::
113 : get_default(detail::kind k) noexcept
114 : {
115 : static constexpr header h[3] = {
116 : fields_tag{},
117 : request_tag{},
118 : response_tag{}};
119 249 : return &h[k];
120 : }
121 :
122 11889 : header::
123 11889 : header(empty v) noexcept
124 11889 : : kind(v.param)
125 : {
126 11889 : }
127 :
128 230 : header::
129 230 : header(detail::kind k) noexcept
130 230 : : header(*get_default(k))
131 : {
132 230 : }
133 :
134 : void
135 74 : header::
136 : swap(header& h) noexcept
137 : {
138 74 : std::swap(cbuf, h.cbuf);
139 74 : std::swap(buf, h.buf);
140 74 : std::swap(cap, h.cap);
141 74 : std::swap(max_cap, h.max_cap);
142 74 : std::swap(size, h.size);
143 74 : std::swap(count, h.count);
144 74 : std::swap(prefix, h.prefix);
145 74 : std::swap(version, h.version);
146 74 : std::swap(md, h.md);
147 74 : switch(kind)
148 : {
149 18 : default:
150 : case detail::kind::fields:
151 18 : break;
152 47 : case detail::kind::request:
153 47 : std::swap(
154 47 : req.method_len, h.req.method_len);
155 47 : std::swap(
156 47 : req.target_len, h.req.target_len);
157 47 : std::swap(req.method, h.req.method);
158 47 : break;
159 9 : case detail::kind::response:
160 9 : std::swap(
161 9 : res.status_int, h.res.status_int);
162 9 : std::swap(res.status, h.res.status);
163 9 : break;
164 : }
165 74 : }
166 :
167 : /* References:
168 :
169 : 6.3. Persistence
170 : https://datatracker.ietf.org/doc/html/rfc7230#section-6.3
171 : */
172 : bool
173 22 : header::
174 : keep_alive() const noexcept
175 : {
176 22 : if(md.payload == payload::error)
177 1 : return false;
178 21 : if( version ==
179 : http_proto::version::http_1_1)
180 : {
181 13 : if(md.connection.close)
182 3 : return false;
183 : }
184 : else
185 : {
186 8 : if(! md.connection.keep_alive)
187 4 : return false;
188 : }
189 : // can't use to_eof in requests
190 14 : BOOST_ASSERT(
191 : kind != detail::kind::request ||
192 : md.payload != payload::to_eof);
193 14 : if(md.payload == payload::to_eof)
194 3 : return false;
195 11 : return true;
196 : }
197 :
198 : //------------------------------------------------
199 :
200 : // return total bytes needed
201 : // to store message of `size`
202 : // bytes and `count` fields.
203 : std::size_t
204 911 : header::
205 : bytes_needed(
206 : std::size_t size,
207 : std::size_t count) noexcept
208 : {
209 : // make sure `size` is big enough
210 : // to hold the largest default buffer:
211 : // "HTTP/1.1 200 OK\r\n\r\n"
212 911 : if( size < 19)
213 173 : size = 19;
214 : static constexpr auto A =
215 : alignof(header::entry);
216 911 : return align_up(size, A) +
217 911 : (count * sizeof(
218 911 : header::entry));
219 : }
220 :
221 : std::size_t
222 9822 : header::
223 : table_space(
224 : std::size_t count) noexcept
225 : {
226 : return count *
227 9822 : sizeof(header::entry);
228 : }
229 :
230 : std::size_t
231 9822 : header::
232 : table_space() const noexcept
233 : {
234 9822 : return table_space(count);
235 : }
236 :
237 : auto
238 15200 : header::
239 : tab() const noexcept ->
240 : table
241 : {
242 15200 : BOOST_ASSERT(cap > 0);
243 15200 : BOOST_ASSERT(buf != nullptr);
244 15200 : return table(buf + cap);
245 : }
246 :
247 : auto
248 857 : header::
249 : tab_() const noexcept ->
250 : entry*
251 : {
252 : return reinterpret_cast<
253 857 : entry*>(buf + cap);
254 : }
255 :
256 : // return true if header cbuf is a default
257 : bool
258 43 : header::
259 : is_default() const noexcept
260 : {
261 43 : return buf == nullptr;
262 : }
263 :
264 : std::size_t
265 4294 : header::
266 : find(
267 : field id) const noexcept
268 : {
269 4294 : if(count == 0)
270 6 : return 0;
271 4288 : std::size_t i = 0;
272 4288 : auto const* p = &tab()[0];
273 4333 : while(i < count)
274 : {
275 4333 : if(p->id == id)
276 4288 : break;
277 45 : ++i;
278 45 : --p;
279 : }
280 4288 : return i;
281 : }
282 :
283 : std::size_t
284 68 : header::
285 : find(
286 : core::string_view name) const noexcept
287 : {
288 68 : if(count == 0)
289 53 : return 0;
290 15 : std::size_t i = 0;
291 15 : auto const* p = &tab()[0];
292 21 : while(i < count)
293 : {
294 : core::string_view s(
295 21 : cbuf + prefix + p->np,
296 21 : p->nn);
297 21 : if(grammar::ci_is_equal(s, name))
298 15 : break;
299 6 : ++i;
300 6 : --p;
301 : }
302 15 : return i;
303 : }
304 :
305 : void
306 30 : header::
307 : copy_table(
308 : void* dest,
309 : std::size_t n) const noexcept
310 : {
311 30 : std::memcpy(
312 : reinterpret_cast<
313 30 : entry*>(dest) - n,
314 : reinterpret_cast<
315 : entry const*>(
316 30 : cbuf + cap) - n,
317 : n * sizeof(entry));
318 30 : }
319 :
320 : void
321 30 : header::
322 : copy_table(
323 : void* dest) const noexcept
324 : {
325 30 : copy_table(dest, count);
326 30 : }
327 :
328 : // assign all the members but
329 : // preserve the allocated memory
330 : void
331 30 : header::
332 : assign_to(
333 : header& dest) const noexcept
334 : {
335 30 : auto const buf_ = dest.buf;
336 30 : auto const cbuf_ = dest.cbuf;
337 30 : auto const cap_ = dest.cap;
338 30 : dest = *this;
339 30 : dest.buf = buf_;
340 30 : dest.cbuf = cbuf_;
341 30 : dest.cap = cap_;
342 30 : }
343 :
344 : //------------------------------------------------
345 : //
346 : // Metadata
347 : //
348 : //------------------------------------------------
349 :
350 : std::size_t
351 0 : header::
352 : maybe_count(
353 : field id) const noexcept
354 : {
355 0 : if(kind == detail::kind::fields)
356 0 : return std::size_t(-1);
357 0 : switch(id)
358 : {
359 0 : case field::connection:
360 0 : return md.connection.count;
361 0 : case field::content_encoding:
362 0 : return md.content_encoding.count;
363 0 : case field::content_length:
364 0 : return md.content_length.count;
365 0 : case field::expect:
366 0 : return md.expect.count;
367 0 : case field::transfer_encoding:
368 0 : return md.transfer_encoding.count;
369 0 : case field::upgrade:
370 0 : return md.upgrade.count;
371 0 : default:
372 0 : break;
373 : }
374 0 : return std::size_t(-1);
375 : }
376 :
377 : bool
378 21 : header::
379 : is_special(
380 : field id) const noexcept
381 : {
382 21 : if(kind == detail::kind::fields)
383 4 : return false;
384 17 : switch(id)
385 : {
386 9 : case field::connection:
387 : case field::content_encoding:
388 : case field::content_length:
389 : case field::expect:
390 : case field::transfer_encoding:
391 : case field::upgrade:
392 9 : return true;
393 8 : default:
394 8 : break;
395 : }
396 8 : return false;
397 : }
398 :
399 : //------------------------------------------------
400 :
401 : // called when the start-line changes
402 : void
403 10593 : header::
404 : on_start_line()
405 : {
406 : // items in both the request-line
407 : // and the status-line can affect
408 : // the payload, for example whether
409 : // or not EOF marks the end of the
410 : // payload.
411 :
412 10593 : update_payload();
413 10593 : }
414 :
415 : // called after a field is inserted
416 : void
417 11752 : header::
418 : on_insert(
419 : field id,
420 : core::string_view v)
421 : {
422 11752 : if(kind == detail::kind::fields)
423 510 : return;
424 11242 : switch(id)
425 : {
426 78 : case field::content_encoding:
427 78 : return on_insert_content_encoding(v);
428 4853 : case field::content_length:
429 4853 : return on_insert_content_length(v);
430 147 : case field::connection:
431 147 : return on_insert_connection(v);
432 47 : case field::expect:
433 47 : return on_insert_expect(v);
434 4275 : case field::transfer_encoding:
435 4275 : return on_insert_transfer_encoding();
436 24 : case field::upgrade:
437 24 : return on_insert_upgrade(v);
438 1818 : default:
439 1818 : break;
440 : }
441 : }
442 :
443 : // called when one field is erased
444 : void
445 40 : header::
446 : on_erase(field id)
447 : {
448 40 : if(kind == detail::kind::fields)
449 3 : return;
450 37 : switch(id)
451 : {
452 9 : case field::connection:
453 9 : return on_erase_connection();
454 0 : case field::content_encoding:
455 0 : return on_erase_content_encoding();
456 4 : case field::content_length:
457 4 : return on_erase_content_length();
458 10 : case field::expect:
459 10 : return on_erase_expect();
460 5 : case field::transfer_encoding:
461 5 : return on_erase_transfer_encoding();
462 4 : case field::upgrade:
463 4 : return on_erase_upgrade();
464 5 : default:
465 5 : break;
466 : }
467 : }
468 :
469 : //------------------------------------------------
470 :
471 : /*
472 : https://datatracker.ietf.org/doc/html/rfc7230#section-6.1
473 : */
474 : void
475 151 : header::
476 : on_insert_connection(
477 : core::string_view v)
478 : {
479 151 : ++md.connection.count;
480 151 : if(md.connection.ec.failed())
481 5 : return;
482 : auto rv = grammar::parse(
483 150 : v, list_rule(token_rule, 1));
484 150 : if(! rv)
485 : {
486 4 : md.connection.ec =
487 8 : BOOST_HTTP_PROTO_ERR(
488 : error::bad_connection);
489 4 : return;
490 : }
491 146 : md.connection.ec = {};
492 303 : for(auto t : *rv)
493 : {
494 157 : if(grammar::ci_is_equal(
495 : t, "close"))
496 107 : md.connection.close = true;
497 50 : else if(grammar::ci_is_equal(
498 : t, "keep-alive"))
499 26 : md.connection.keep_alive = true;
500 24 : else if(grammar::ci_is_equal(
501 : t, "upgrade"))
502 19 : md.connection.upgrade = true;
503 : }
504 150 : }
505 :
506 : void
507 4854 : header::
508 : on_insert_content_length(
509 : core::string_view v)
510 : {
511 : static
512 : constexpr
513 : grammar::unsigned_rule<
514 : std::uint64_t> num_rule{};
515 :
516 4854 : ++md.content_length.count;
517 4854 : if(md.content_length.ec.failed())
518 4671 : return;
519 : auto rv =
520 4852 : grammar::parse(v, num_rule);
521 4852 : if(! rv)
522 : {
523 : // parse failure
524 5 : md.content_length.ec =
525 10 : BOOST_HTTP_PROTO_ERR(
526 : error::bad_content_length);
527 5 : md.content_length.value = 0;
528 5 : update_payload();
529 5 : return;
530 : }
531 4847 : if(md.content_length.count == 1)
532 : {
533 : // one value
534 4657 : md.content_length.ec = {};
535 4657 : md.content_length.value = *rv;
536 4657 : update_payload();
537 4657 : return;
538 : }
539 190 : if(*rv == md.content_length.value)
540 : {
541 : // ok: duplicate value
542 7 : return;
543 : }
544 : // bad: different values
545 183 : md.content_length.ec =
546 366 : BOOST_HTTP_PROTO_ERR(
547 : error::multiple_content_length);
548 183 : md.content_length.value = 0;
549 183 : update_payload();
550 : }
551 :
552 : void
553 53 : header::
554 : on_insert_expect(
555 : core::string_view v)
556 : {
557 53 : ++md.expect.count;
558 53 : if(kind != detail::kind::request)
559 8 : return;
560 45 : if(md.expect.ec.failed())
561 4 : return;
562 : // VFALCO Should we allow duplicate
563 : // Expect fields that have 100-continue?
564 73 : if( md.expect.count > 1 ||
565 73 : ! grammar::ci_is_equal(v,
566 : "100-continue"))
567 : {
568 19 : md.expect.ec =
569 38 : BOOST_HTTP_PROTO_ERR(
570 : error::bad_expect);
571 19 : md.expect.is_100_continue = false;
572 19 : return;
573 : }
574 22 : md.expect.is_100_continue = true;
575 : }
576 :
577 : void
578 4278 : header::
579 : on_insert_transfer_encoding()
580 : {
581 4278 : ++md.transfer_encoding.count;
582 4278 : if(md.transfer_encoding.ec.failed())
583 5 : return;
584 4273 : auto const n =
585 : md.transfer_encoding.count;
586 4273 : md.transfer_encoding = {};
587 4273 : md.transfer_encoding.count = n;
588 4273 : for(auto s :
589 : fields_view_base::subrange(
590 12840 : this, find(field::transfer_encoding)))
591 : {
592 : auto rv = grammar::parse(
593 4305 : s, transfer_encoding_rule);
594 4305 : if(! rv)
595 : {
596 : // parse error
597 4 : md.transfer_encoding.ec =
598 8 : BOOST_HTTP_PROTO_ERR(
599 : error::bad_transfer_encoding);
600 4 : md.transfer_encoding.codings = 0;
601 4 : md.transfer_encoding.is_chunked = false;
602 4 : update_payload();
603 4 : return;
604 : }
605 4301 : md.transfer_encoding.codings += rv->size();
606 8604 : for(auto t : *rv)
607 : {
608 4310 : auto& mte = md.transfer_encoding;
609 :
610 4310 : if(! mte.is_chunked )
611 : {
612 4306 : if( t.id == transfer_encoding::chunked )
613 : {
614 4205 : mte.is_chunked = true;
615 4205 : continue;
616 : }
617 :
618 101 : auto b =
619 101 : mte.encoding ==
620 : http_proto::encoding::identity;
621 :
622 101 : if( t.id == transfer_encoding::deflate )
623 37 : mte.encoding = http_proto::encoding::deflate;
624 :
625 101 : if( t.id == transfer_encoding::gzip )
626 43 : mte.encoding = http_proto::encoding::gzip;
627 :
628 101 : if( b )
629 98 : continue;
630 : }
631 7 : if(t.id == transfer_encoding::chunked)
632 : {
633 : // chunked appears twice
634 2 : md.transfer_encoding.ec =
635 4 : BOOST_HTTP_PROTO_ERR(
636 : error::bad_transfer_encoding);
637 2 : md.transfer_encoding.codings = 0;
638 2 : md.transfer_encoding.is_chunked = false;
639 2 : md.transfer_encoding.encoding =
640 : http_proto::encoding::identity;
641 2 : update_payload();
642 2 : return;
643 : }
644 : // chunked must be last
645 5 : md.transfer_encoding.ec =
646 10 : BOOST_HTTP_PROTO_ERR(
647 : error::bad_transfer_encoding);
648 5 : md.transfer_encoding.codings = 0;
649 5 : md.transfer_encoding.is_chunked = false;
650 5 : md.transfer_encoding.encoding =
651 : http_proto::encoding::identity;
652 5 : update_payload();
653 5 : return;
654 8618 : }
655 4305 : }
656 4262 : update_payload();
657 : }
658 :
659 : void
660 78 : header::
661 : on_insert_content_encoding(
662 : core::string_view v)
663 : {
664 78 : ++md.content_encoding.count;
665 78 : if( md.content_encoding.ec.failed() )
666 3 : return;
667 :
668 : auto rv = grammar::parse(
669 78 : v, list_rule(token_rule, 1));
670 78 : if( !rv )
671 : {
672 1 : md.content_encoding.ec =
673 2 : BOOST_HTTP_PROTO_ERR(
674 : error::bad_content_encoding);
675 1 : return;
676 : }
677 :
678 153 : if( rv->size() > 1 ||
679 76 : md.content_encoding.count > 1)
680 : {
681 2 : md.content_encoding.encoding =
682 : encoding::unsupported;
683 2 : return;
684 : }
685 :
686 75 : if( grammar::ci_is_equal(*(rv->begin()),
687 : "deflate") )
688 : {
689 37 : md.content_encoding.encoding =
690 : encoding::deflate;
691 : }
692 38 : else if( grammar::ci_is_equal(*(rv->begin()),
693 : "gzip") )
694 : {
695 38 : md.content_encoding.encoding =
696 : encoding::gzip;
697 : }
698 : else
699 : {
700 0 : md.content_encoding.encoding =
701 : encoding::unsupported;
702 : }
703 78 : }
704 :
705 : void
706 26 : header::
707 : on_insert_upgrade(
708 : core::string_view v)
709 : {
710 26 : ++md.upgrade.count;
711 26 : if(md.upgrade.ec.failed())
712 5 : return;
713 25 : if( version !=
714 : http_proto::version::http_1_1)
715 : {
716 1 : md.upgrade.ec =
717 2 : BOOST_HTTP_PROTO_ERR(
718 : error::bad_upgrade);
719 1 : md.upgrade.websocket = false;
720 1 : return;
721 : }
722 : auto rv = grammar::parse(
723 24 : v, upgrade_rule);
724 24 : if(! rv)
725 : {
726 3 : md.upgrade.ec =
727 6 : BOOST_HTTP_PROTO_ERR(
728 : error::bad_upgrade);
729 3 : md.upgrade.websocket = false;
730 3 : return;
731 : }
732 21 : if(! md.upgrade.websocket)
733 : {
734 23 : for(auto t : *rv)
735 : {
736 16 : if( grammar::ci_is_equal(
737 26 : t.name, "websocket") &&
738 10 : t.version.empty())
739 : {
740 9 : md.upgrade.websocket = true;
741 9 : break;
742 : }
743 : }
744 : }
745 24 : }
746 :
747 : //------------------------------------------------
748 :
749 : void
750 9 : header::
751 : on_erase_connection()
752 : {
753 9 : BOOST_ASSERT(
754 : md.connection.count > 0);
755 : // reset and re-insert
756 9 : auto n = md.connection.count - 1;
757 9 : auto const p = cbuf + prefix;
758 9 : auto const* e = &tab()[0];
759 9 : md.connection = {};
760 14 : while(n > 0)
761 : {
762 5 : if(e->id == field::connection)
763 4 : on_insert_connection(
764 : core::string_view(
765 4 : p + e->vp, e->vn));
766 5 : --n;
767 5 : --e;
768 : }
769 9 : }
770 :
771 : void
772 4 : header::
773 : on_erase_content_length()
774 : {
775 4 : BOOST_ASSERT(
776 : md.content_length.count > 0);
777 4 : --md.content_length.count;
778 4 : if(md.content_length.count == 0)
779 : {
780 : // no Content-Length
781 1 : md.content_length = {};
782 1 : update_payload();
783 1 : return;
784 : }
785 3 : if(! md.content_length.ec.failed())
786 : {
787 : // removing a duplicate value
788 2 : return;
789 : }
790 : // reset and re-insert
791 1 : auto n = md.content_length.count;
792 1 : auto const p = cbuf + prefix;
793 1 : auto const* e = &tab()[0];
794 1 : md.content_length = {};
795 2 : while(n > 0)
796 : {
797 1 : if(e->id == field::content_length)
798 1 : on_insert_content_length(
799 : core::string_view(
800 1 : p + e->vp, e->vn));
801 1 : --n;
802 1 : --e;
803 : }
804 1 : update_payload();
805 : }
806 :
807 : void
808 10 : header::
809 : on_erase_expect()
810 : {
811 10 : BOOST_ASSERT(
812 : md.expect.count > 0);
813 10 : --md.expect.count;
814 10 : if(kind != detail::kind::request)
815 1 : return;
816 9 : if(md.expect.count == 0)
817 : {
818 : // no Expect
819 3 : md.expect = {};
820 3 : return;
821 : }
822 : // VFALCO This should be uncommented
823 : // if we want to allow multiple Expect
824 : // fields with the value 100-continue
825 : /*
826 : if(! md.expect.ec.failed())
827 : return;
828 : */
829 : // reset and re-insert
830 6 : auto n = count;
831 6 : auto const p = cbuf + prefix;
832 6 : auto const* e = &tab()[0];
833 6 : md.expect = {};
834 19 : while(n > 0)
835 : {
836 13 : if(e->id == field::expect)
837 6 : on_insert_expect(
838 : core::string_view(
839 6 : p + e->vp, e->vn));
840 13 : --n;
841 13 : --e;
842 : }
843 : }
844 :
845 : void
846 5 : header::
847 : on_erase_transfer_encoding()
848 : {
849 5 : BOOST_ASSERT(
850 : md.transfer_encoding.count > 0);
851 5 : --md.transfer_encoding.count;
852 5 : if(md.transfer_encoding.count == 0)
853 : {
854 : // no Transfer-Encoding
855 2 : md.transfer_encoding = {};
856 2 : update_payload();
857 2 : return;
858 : }
859 : // re-insert everything
860 3 : --md.transfer_encoding.count;
861 3 : on_insert_transfer_encoding();
862 : }
863 :
864 : void
865 0 : header::
866 : on_erase_content_encoding()
867 : {
868 0 : BOOST_ASSERT(
869 : md.content_encoding.count > 0);
870 0 : --md.content_encoding.count;
871 0 : if(md.content_encoding.count == 0)
872 : {
873 : // no Content-Encoding
874 0 : md.content_encoding = {};
875 0 : return;
876 : }
877 : // re-insert everything
878 0 : --md.content_encoding.count;
879 : // TODO
880 : // on_insert_content_encoding();
881 : }
882 :
883 : // called when Upgrade is erased
884 : void
885 4 : header::
886 : on_erase_upgrade()
887 : {
888 4 : BOOST_ASSERT(
889 : md.upgrade.count > 0);
890 4 : --md.upgrade.count;
891 4 : if(md.upgrade.count == 0)
892 : {
893 : // no Upgrade
894 2 : md.upgrade = {};
895 2 : return;
896 : }
897 : // reset and re-insert
898 2 : auto n = md.upgrade.count;
899 2 : auto const p = cbuf + prefix;
900 2 : auto const* e = &tab()[0];
901 2 : md.upgrade = {};
902 4 : while(n > 0)
903 : {
904 2 : if(e->id == field::upgrade)
905 2 : on_insert_upgrade(
906 : core::string_view(
907 2 : p + e->vp, e->vn));
908 2 : --n;
909 2 : --e;
910 : }
911 : }
912 :
913 : //------------------------------------------------
914 :
915 : // called when all fields with id are removed
916 : void
917 60 : header::
918 : on_erase_all(
919 : field id)
920 : {
921 60 : if(kind == detail::kind::fields)
922 17 : return;
923 43 : switch(id)
924 : {
925 3 : case field::connection:
926 3 : md.connection = {};
927 3 : return;
928 :
929 2 : case field::content_length:
930 2 : md.content_length = {};
931 2 : update_payload();
932 2 : return;
933 :
934 5 : case field::expect:
935 5 : md.expect = {};
936 5 : update_payload();
937 5 : return;
938 :
939 1 : case field::transfer_encoding:
940 1 : md.transfer_encoding = {};
941 1 : update_payload();
942 1 : return;
943 :
944 1 : case field::upgrade:
945 1 : md.upgrade = {};
946 1 : return;
947 :
948 31 : default:
949 31 : break;
950 : }
951 : }
952 :
953 : //------------------------------------------------
954 :
955 : /* References:
956 :
957 : 3.3. Message Body
958 : https://datatracker.ietf.org/doc/html/rfc7230#section-3.3
959 :
960 : 3.3.1. Transfer-Encoding
961 : https://datatracker.ietf.org/doc/html/rfc7230#section-3.3.1
962 :
963 : 3.3.2. Content-Length
964 : https://datatracker.ietf.org/doc/html/rfc7230#section-3.3.2
965 : */
966 : void
967 19723 : header::
968 : update_payload() noexcept
969 : {
970 19723 : BOOST_ASSERT(kind !=
971 : detail::kind::fields);
972 19723 : if(md.payload_override)
973 : {
974 : // e.g. response to
975 : // a HEAD request
976 0 : return;
977 : }
978 :
979 : /* If there is an error in either Content-Length
980 : or Transfer-Encoding, then the payload is
981 : undefined. Clients should probably close the
982 : connection. Servers can send a Bad Request
983 : and avoid reading any payload bytes.
984 : */
985 19723 : if(md.content_length.ec.failed())
986 : {
987 : // invalid Content-Length
988 188 : md.payload = payload::error;
989 188 : md.payload_size = 0;
990 188 : return;
991 : }
992 19535 : if(md.transfer_encoding.ec.failed())
993 : {
994 : // invalid Transfer-Encoding
995 11 : md.payload = payload::error;
996 11 : md.payload_size = 0;
997 11 : return;
998 : }
999 :
1000 : /* A sender MUST NOT send a Content-Length
1001 : header field in any message that contains
1002 : a Transfer-Encoding header field.
1003 : https://datatracker.ietf.org/doc/html/rfc7230#section-3.3.2
1004 : */
1005 19524 : if( md.content_length.count > 0 &&
1006 4661 : md.transfer_encoding.count > 0)
1007 : {
1008 3 : md.payload = payload::error;
1009 3 : md.payload_size = 0;
1010 3 : return;
1011 : }
1012 :
1013 19521 : if(kind == detail::kind::response)
1014 1323 : goto do_response;
1015 :
1016 : //--------------------------------------------
1017 :
1018 : /* The presence of a message body in a
1019 : request is signaled by a Content-Length
1020 : or Transfer-Encoding header field. Request
1021 : message framing is independent of method
1022 : semantics, even if the method does not
1023 : define any use for a message body.
1024 : */
1025 18198 : if(md.content_length.count > 0)
1026 : {
1027 4419 : if(md.content_length.value > 0)
1028 : {
1029 : // non-zero Content-Length
1030 4392 : md.payload = payload::size;
1031 4392 : md.payload_size = md.content_length.value;
1032 4392 : return;
1033 : }
1034 : // Content-Length: 0
1035 27 : md.payload = payload::none;
1036 27 : md.payload_size = 0;
1037 27 : return;
1038 : }
1039 13779 : if(md.transfer_encoding.is_chunked)
1040 : {
1041 : // chunked
1042 4010 : md.payload = payload::chunked;
1043 4010 : md.payload_size = 0;
1044 4010 : return;
1045 : }
1046 : // no payload
1047 9769 : md.payload = payload::none;
1048 9769 : md.payload_size = 0;
1049 9769 : return;
1050 :
1051 : //--------------------------------------------
1052 1323 : do_response:
1053 :
1054 1323 : if( res.status_int / 100 == 1 || // 1xx e.g. Continue
1055 1313 : res.status_int == 204 || // No Content
1056 1311 : res.status_int == 304) // Not Modified
1057 : {
1058 : /* The correctness of any Content-Length
1059 : here is defined by the particular
1060 : resource, and cannot be determined
1061 : here. In any case there is no payload.
1062 : */
1063 14 : md.payload = payload::none;
1064 14 : md.payload_size = 0;
1065 14 : return;
1066 : }
1067 1309 : if(md.content_length.count > 0)
1068 : {
1069 236 : if(md.content_length.value > 0)
1070 : {
1071 : // Content-Length > 0
1072 223 : md.payload = payload::size;
1073 223 : md.payload_size = md.content_length.value;
1074 223 : return;
1075 : }
1076 : // Content-Length: 0
1077 13 : md.payload = payload::none;
1078 13 : md.payload_size = 0;
1079 13 : return;
1080 : }
1081 1073 : if(md.transfer_encoding.is_chunked)
1082 : {
1083 : // chunked
1084 188 : md.payload = payload::chunked;
1085 188 : md.payload_size = 0;
1086 188 : return;
1087 : }
1088 :
1089 : // eof needed
1090 885 : md.payload = payload::to_eof;
1091 885 : md.payload_size = 0;
1092 : }
1093 :
1094 : //------------------------------------------------
1095 :
1096 : std::size_t
1097 541 : header::
1098 : count_crlf(
1099 : core::string_view s) noexcept
1100 : {
1101 541 : auto it = s.data();
1102 541 : auto len = s.size();
1103 541 : std::size_t n = 0;
1104 19037 : while(len >= 2)
1105 : {
1106 18496 : if( it[0] == '\r' &&
1107 1730 : it[1] != '\r')
1108 : {
1109 1730 : if(it[1] == '\n')
1110 1730 : n++;
1111 1730 : it += 2;
1112 1730 : len -= 2;
1113 : }
1114 : else
1115 : {
1116 16766 : it++;
1117 16766 : len--;
1118 : }
1119 : }
1120 541 : return n;
1121 : }
1122 :
1123 : static
1124 : void
1125 12675 : parse_start_line(
1126 : header& h,
1127 : header_limits const& lim,
1128 : std::size_t new_size,
1129 : system::error_code& ec) noexcept
1130 : {
1131 12675 : BOOST_ASSERT(h.size == 0);
1132 12675 : BOOST_ASSERT(h.prefix == 0);
1133 12675 : BOOST_ASSERT(h.cbuf != nullptr);
1134 12675 : BOOST_ASSERT(
1135 : h.kind != detail::kind::fields);
1136 :
1137 12675 : auto const it0 = h.cbuf;
1138 12675 : auto const end = it0 + new_size;
1139 12675 : char const* it = it0;
1140 12675 : if( new_size > lim.max_start_line)
1141 36 : new_size = lim.max_start_line;
1142 12675 : if(h.kind == detail::kind::request)
1143 : {
1144 : auto rv = grammar::parse(
1145 11622 : it, end, request_line_rule);
1146 11622 : if(! rv)
1147 : {
1148 1892 : ec = rv.error();
1149 3784 : if( ec == grammar::error::need_more &&
1150 1892 : new_size == lim.max_start_line)
1151 0 : ec = BOOST_HTTP_PROTO_ERR(
1152 : error::start_line_limit);
1153 1892 : return;
1154 : }
1155 : // method
1156 9730 : auto sm = std::get<0>(*rv);
1157 9730 : h.req.method = string_to_method(sm);
1158 9730 : h.req.method_len =
1159 9730 : static_cast<offset_type>(sm.size());
1160 : // target
1161 9730 : auto st = std::get<1>(*rv);
1162 9730 : h.req.target_len =
1163 9730 : static_cast<offset_type>(st.size());
1164 : // version
1165 9730 : switch(std::get<2>(*rv))
1166 : {
1167 20 : case 10:
1168 20 : h.version =
1169 : http_proto::version::http_1_0;
1170 20 : break;
1171 9710 : case 11:
1172 9710 : h.version =
1173 : http_proto::version::http_1_1;
1174 9710 : break;
1175 0 : default:
1176 : {
1177 0 : ec = BOOST_HTTP_PROTO_ERR(
1178 : error::bad_version);
1179 0 : return;
1180 : }
1181 : }
1182 : }
1183 : else
1184 : {
1185 : auto rv = grammar::parse(
1186 1053 : it, end, status_line_rule);
1187 1053 : if(! rv)
1188 : {
1189 224 : ec = rv.error();
1190 448 : if( ec == grammar::error::need_more &&
1191 224 : new_size == lim.max_start_line)
1192 0 : ec = BOOST_HTTP_PROTO_ERR(
1193 : error::start_line_limit);
1194 224 : return;
1195 : }
1196 : // version
1197 829 : switch(std::get<0>(*rv))
1198 : {
1199 5 : case 10:
1200 5 : h.version =
1201 : http_proto::version::http_1_0;
1202 5 : break;
1203 824 : case 11:
1204 824 : h.version =
1205 : http_proto::version::http_1_1;
1206 824 : break;
1207 0 : default:
1208 : {
1209 0 : ec = BOOST_HTTP_PROTO_ERR(
1210 : error::bad_version);
1211 0 : return;
1212 : }
1213 : }
1214 : // status-code
1215 829 : h.res.status_int =
1216 : static_cast<unsigned short>(
1217 829 : std::get<1>(*rv).v);
1218 829 : h.res.status = std::get<1>(*rv).st;
1219 : }
1220 10559 : h.prefix = static_cast<offset_type>(it - it0);
1221 10559 : h.size = h.prefix;
1222 10559 : h.on_start_line();
1223 : }
1224 :
1225 : // returns: true if we added a field
1226 : static
1227 : void
1228 24288 : parse_field(
1229 : header& h,
1230 : header_limits const& lim,
1231 : std::size_t new_size,
1232 : system::error_code& ec) noexcept
1233 : {
1234 24288 : if( new_size > lim.max_field)
1235 96 : new_size = lim.max_field;
1236 24288 : auto const it0 = h.cbuf + h.size;
1237 24288 : auto const end = h.cbuf + new_size;
1238 24288 : char const* it = it0;
1239 24288 : auto rv = grammar::parse(
1240 : it, end, field_rule);
1241 24288 : if(rv.has_error())
1242 : {
1243 12815 : ec = rv.error();
1244 12815 : if(ec == grammar::error::end_of_range)
1245 : {
1246 : // final CRLF
1247 10540 : h.size = static_cast<
1248 10540 : offset_type>(it - h.cbuf);
1249 12815 : return;
1250 : }
1251 4291 : if( ec == grammar::error::need_more &&
1252 2016 : new_size == lim.max_field)
1253 : {
1254 0 : ec = BOOST_HTTP_PROTO_ERR(
1255 : error::field_size_limit);
1256 : }
1257 2275 : return;
1258 : }
1259 11473 : if(h.count >= lim.max_fields)
1260 : {
1261 0 : ec = BOOST_HTTP_PROTO_ERR(
1262 : error::fields_limit);
1263 0 : return;
1264 : }
1265 11473 : if(rv->has_obs_fold)
1266 : {
1267 : // obs fold not allowed in test views
1268 210 : BOOST_ASSERT(h.buf != nullptr);
1269 210 : remove_obs_fold(h.buf + h.size, it);
1270 : }
1271 11473 : auto id = string_to_field(rv->name);
1272 11473 : h.size = static_cast<offset_type>(it - h.cbuf);
1273 :
1274 : // add field table entry
1275 11473 : if(h.buf != nullptr)
1276 : {
1277 22946 : auto& e = header::table(
1278 11473 : h.buf + h.cap)[h.count];
1279 11473 : auto const base =
1280 11473 : h.buf + h.prefix;
1281 11473 : e.np = static_cast<offset_type>(
1282 11473 : rv->name.data() - base);
1283 11473 : e.nn = static_cast<offset_type>(
1284 11473 : rv->name.size());
1285 11473 : e.vp = static_cast<offset_type>(
1286 11473 : rv->value.data() - base);
1287 11473 : e.vn = static_cast<offset_type>(
1288 11473 : rv->value.size());
1289 11473 : e.id = id;
1290 : }
1291 11473 : ++h.count;
1292 11473 : h.on_insert(id, rv->value);
1293 11473 : ec = {};
1294 : }
1295 :
1296 : void
1297 14931 : header::
1298 : parse(
1299 : std::size_t new_size,
1300 : header_limits const& lim,
1301 : system::error_code& ec) noexcept
1302 : {
1303 14931 : if( new_size > lim.max_size)
1304 36 : new_size = lim.max_size;
1305 14931 : if( this->prefix == 0 &&
1306 12915 : this->kind !=
1307 : detail::kind::fields)
1308 : {
1309 12675 : parse_start_line(
1310 : *this, lim, new_size, ec);
1311 12675 : if(ec.failed())
1312 : {
1313 4232 : if( ec == grammar::error::need_more &&
1314 2116 : new_size == lim.max_fields)
1315 : {
1316 0 : ec = BOOST_HTTP_PROTO_ERR(
1317 : error::headers_limit);
1318 : }
1319 2116 : return;
1320 : }
1321 : }
1322 : for(;;)
1323 : {
1324 24288 : parse_field(
1325 : *this, lim, new_size, ec);
1326 24288 : if(ec.failed())
1327 : {
1328 14831 : if( ec == grammar::error::need_more &&
1329 2016 : new_size == lim.max_size)
1330 : {
1331 0 : ec = BOOST_HTTP_PROTO_ERR(
1332 : error::headers_limit);
1333 0 : return;
1334 : }
1335 12815 : break;
1336 : }
1337 11473 : }
1338 12815 : if(ec == grammar::error::end_of_range)
1339 10540 : ec = {};
1340 : }
1341 :
1342 : } // detail
1343 : } // http_proto
1344 : } // boost
|