@@ -385,7 +385,7 @@ impl Http1Transaction for Server {
385
385
} } ;
386
386
}
387
387
388
- ' headers : for ( opt_name, value) in msg. head . headers . drain ( ) {
388
+ for ( opt_name, value) in msg. head . headers . drain ( ) {
389
389
if let Some ( n) = opt_name {
390
390
cur_name = Some ( n) ;
391
391
handle_is_name_written ! ( ) ;
@@ -399,6 +399,14 @@ impl Http1Transaction for Server {
399
399
rewind ( dst) ;
400
400
return Err ( crate :: Error :: new_user_header ( ) ) ;
401
401
}
402
+ // Can this response even contain Content-Length?
403
+ if !Server :: can_have_body ( msg. req_method , msg. head . subject ) {
404
+ warn ! (
405
+ "illegal content-length ({:?}) seen for {:?} response to {:?} request" ,
406
+ value, msg. head. subject, msg. req_method
407
+ ) ;
408
+ continue ;
409
+ }
402
410
match msg. body {
403
411
Some ( BodyLength :: Known ( known_len) ) => {
404
412
// The HttpBody claims to know a length, and
@@ -421,13 +429,15 @@ impl Http1Transaction for Server {
421
429
}
422
430
423
431
if !is_name_written {
424
- encoder = Encoder :: length ( known_len) ;
432
+ if !Server :: imply_body_only ( msg. req_method , msg. head . subject ) {
433
+ encoder = Encoder :: length ( known_len) ;
434
+ }
425
435
extend ( dst, b"content-length: " ) ;
426
436
extend ( dst, value. as_bytes ( ) ) ;
427
437
wrote_len = true ;
428
438
is_name_written = true ;
429
439
}
430
- continue ' headers ;
440
+ continue ;
431
441
}
432
442
Some ( BodyLength :: Unknown ) => {
433
443
// The HttpBody impl didn't know how long the
@@ -446,16 +456,18 @@ impl Http1Transaction for Server {
446
456
return Err ( crate :: Error :: new_user_header ( ) ) ;
447
457
}
448
458
debug_assert ! ( is_name_written) ;
449
- continue ' headers ;
459
+ continue ;
450
460
} else {
451
461
// we haven't written content-length yet!
452
- encoder = Encoder :: length ( len) ;
462
+ if !Server :: imply_body_only ( msg. req_method , msg. head . subject ) {
463
+ encoder = Encoder :: length ( len) ;
464
+ }
453
465
extend ( dst, b"content-length: " ) ;
454
466
extend ( dst, value. as_bytes ( ) ) ;
455
467
wrote_len = true ;
456
468
is_name_written = true ;
457
469
prev_con_len = Some ( len) ;
458
- continue ' headers ;
470
+ continue ;
459
471
}
460
472
} else {
461
473
warn ! ( "illegal Content-Length value: {:?}" , value) ;
@@ -464,13 +476,13 @@ impl Http1Transaction for Server {
464
476
}
465
477
}
466
478
None => {
467
- // We have no body to actually send,
468
- // but the headers claim a content-length.
469
- // There's only 2 ways this makes sense:
479
+ // We have no body to actually send, but the headers claim a
480
+ // content-length. There are 3 ways this makes sense:
470
481
//
471
482
// - The header says the length is `0`.
483
+ // - The response is NOT_MODIFIED
472
484
// - This is a response to a `HEAD` request.
473
- if msg. req_method == & Some ( Method :: HEAD ) {
485
+ if Server :: imply_body_only ( msg. req_method , msg . head . subject ) {
474
486
debug_assert_eq ! ( encoder, Encoder :: length( 0 ) ) ;
475
487
} else {
476
488
if value. as_bytes ( ) != b"0" {
@@ -479,7 +491,7 @@ impl Http1Transaction for Server {
479
491
value
480
492
) ;
481
493
}
482
- continue ' headers ;
494
+ continue ;
483
495
}
484
496
}
485
497
}
@@ -492,9 +504,13 @@ impl Http1Transaction for Server {
492
504
return Err ( crate :: Error :: new_user_header ( ) ) ;
493
505
}
494
506
// check that we actually can send a chunked body...
495
- if msg. head . version == Version :: HTTP_10
496
- || !Server :: can_chunked ( msg. req_method , msg. head . subject )
497
- {
507
+ if msg. head . version == Version :: HTTP_10 {
508
+ continue ;
509
+ } else if !Server :: can_have_body ( msg. req_method , msg. head . subject ) {
510
+ warn ! (
511
+ "illegal transfer-encoding ({:?}) seen for {:?} response to {:?} request" ,
512
+ value, msg. head. subject, msg. req_method
513
+ ) ;
498
514
continue ;
499
515
}
500
516
wrote_len = true ;
@@ -503,15 +519,17 @@ impl Http1Transaction for Server {
503
519
must_write_chunked = !headers:: is_chunked_ ( & value) ;
504
520
505
521
if !is_name_written {
506
- encoder = Encoder :: chunked ( ) ;
522
+ if !Server :: imply_body_only ( msg. req_method , msg. head . subject ) {
523
+ encoder = Encoder :: chunked ( ) ;
524
+ }
507
525
is_name_written = true ;
508
526
extend ( dst, b"transfer-encoding: " ) ;
509
527
extend ( dst, value. as_bytes ( ) ) ;
510
528
} else {
511
529
extend ( dst, b", " ) ;
512
530
extend ( dst, value. as_bytes ( ) ) ;
513
531
}
514
- continue ' headers ;
532
+ continue ;
515
533
}
516
534
header:: CONNECTION => {
517
535
if !is_last && headers:: connection_close ( & value) {
@@ -525,7 +543,7 @@ impl Http1Transaction for Server {
525
543
extend ( dst, b", " ) ;
526
544
extend ( dst, value. as_bytes ( ) ) ;
527
545
}
528
- continue ' headers ;
546
+ continue ;
529
547
}
530
548
header:: DATE => {
531
549
wrote_date = true ;
@@ -549,44 +567,50 @@ impl Http1Transaction for Server {
549
567
550
568
handle_is_name_written ! ( ) ;
551
569
552
- if !wrote_len {
553
- encoder = match msg. body {
554
- Some ( BodyLength :: Unknown ) => {
555
- if msg. head . version == Version :: HTTP_10
556
- || !Server :: can_chunked ( msg. req_method , msg. head . subject )
557
- {
558
- Encoder :: close_delimited ( )
559
- } else {
560
- extend ( dst, b"transfer-encoding: chunked\r \n " ) ;
561
- Encoder :: chunked ( )
562
- }
570
+ // We should provide either a Content-Length or Transfer-Encoding header based on what we
571
+ // know about the Body, unless:
572
+ // - the request was HEAD and the body is Body::empty(), because a service may simply not
573
+ // provide generate the real body for a HEAD request.
574
+ // - the response is a 304, like a HEAD, may imply things about a particular body, but a
575
+ // service probably did not provide the full body in its Response.
576
+ // - the response must neither contain nor imply a body.
577
+ if !wrote_len && Server :: can_have_body ( msg. req_method , msg. head . subject ) {
578
+ match msg. body {
579
+ Some ( BodyLength :: Known ( 0 ) ) => extend ( dst, b"content-length: 0\r \n " ) ,
580
+ Some ( BodyLength :: Known ( len) ) => {
581
+ extend ( dst, b"content-length: " ) ;
582
+ let _ = :: itoa:: write ( & mut dst, len) ;
583
+ extend ( dst, b"\r \n " ) ;
563
584
}
564
- None | Some ( BodyLength :: Known ( 0 ) ) => {
565
- if msg. head . subject != StatusCode :: NOT_MODIFIED {
585
+ None => {
586
+ if ! Server :: imply_body_only ( msg. req_method , msg . head . subject ) {
566
587
extend ( dst, b"content-length: 0\r \n " ) ;
567
588
}
568
- Encoder :: length ( 0 )
569
589
}
570
- Some ( BodyLength :: Known ( len) ) => {
571
- if msg. head . subject == StatusCode :: NOT_MODIFIED {
572
- Encoder :: length ( 0 )
573
- } else {
574
- extend ( dst, b"content-length: " ) ;
575
- let _ = :: itoa:: write ( & mut dst, len) ;
576
- extend ( dst, b"\r \n " ) ;
577
- Encoder :: length ( len)
590
+ Some ( BodyLength :: Unknown ) => {
591
+ if msg. head . version != Version :: HTTP_10 {
592
+ extend ( dst, b"transfer-encoding: chunked\r \n " ) ;
578
593
}
579
594
}
580
595
} ;
596
+
597
+ if !Server :: imply_body_only ( msg. req_method , msg. head . subject ) {
598
+ encoder = match msg. body {
599
+ Some ( BodyLength :: Unknown ) if msg. head . version == Version :: HTTP_10 => {
600
+ Encoder :: close_delimited ( )
601
+ }
602
+ Some ( BodyLength :: Unknown ) => Encoder :: chunked ( ) ,
603
+ Some ( BodyLength :: Known ( len) ) => Encoder :: length ( len) ,
604
+ None => Encoder :: length ( 0 ) ,
605
+ } ;
606
+ }
581
607
}
582
608
583
- if !Server :: can_have_body ( msg. req_method , msg. head . subject ) {
584
- trace ! (
585
- "server body forced to 0; method={:?}, status={:?}" ,
586
- msg. req_method,
587
- msg. head. subject
588
- ) ;
589
- encoder = Encoder :: length ( 0 ) ;
609
+ #[ cfg( debug_assertions) ]
610
+ if Server :: imply_body_only ( msg. req_method , msg. head . subject )
611
+ || !Server :: can_have_body ( msg. req_method , msg. head . subject )
612
+ {
613
+ assert_eq ! ( encoder, Encoder :: length( 0 ) ) ;
590
614
}
591
615
592
616
// cached date is much faster than formatting every request
@@ -631,24 +655,19 @@ impl Http1Transaction for Server {
631
655
#[ cfg( feature = "server" ) ]
632
656
impl Server {
633
657
fn can_have_body ( method : & Option < Method > , status : StatusCode ) -> bool {
634
- Server :: can_chunked ( method, status)
635
- }
636
-
637
- fn can_chunked ( method : & Option < Method > , status : StatusCode ) -> bool {
638
- if method == & Some ( Method :: HEAD ) || method == & Some ( Method :: CONNECT ) && status. is_success ( )
658
+ if method == & Some ( Method :: CONNECT ) && status. is_success ( )
659
+ || status == StatusCode :: NO_CONTENT
660
+ || status. is_informational ( )
639
661
{
640
662
false
641
663
} else {
642
- match status {
643
- // TODO: support for 1xx codes needs improvement everywhere
644
- // would be 100...199 => false
645
- StatusCode :: SWITCHING_PROTOCOLS
646
- | StatusCode :: NO_CONTENT
647
- | StatusCode :: NOT_MODIFIED => false ,
648
- _ => true ,
649
- }
664
+ true
650
665
}
651
666
}
667
+
668
+ fn imply_body_only ( method : & Option < Method > , status : StatusCode ) -> bool {
669
+ method == & Some ( Method :: HEAD ) || status == StatusCode :: NOT_MODIFIED
670
+ }
652
671
}
653
672
654
673
#[ cfg( feature = "client" ) ]
0 commit comments