@@ -5,8 +5,6 @@ use std::fmt;
5
5
6
6
use bytes:: Bytes ;
7
7
use futures_channel:: mpsc;
8
- #[ cfg( any( feature = "http1" , feature = "http2" ) ) ]
9
- #[ cfg( feature = "client" ) ]
10
8
use futures_channel:: oneshot;
11
9
use futures_core:: Stream ; // for mpsc::Receiver
12
10
#[ cfg( feature = "stream" ) ]
@@ -17,14 +15,16 @@ use http_body::{Body as HttpBody, SizeHint};
17
15
use super :: DecodedLength ;
18
16
#[ cfg( feature = "stream" ) ]
19
17
use crate :: common:: sync_wrapper:: SyncWrapper ;
20
- use crate :: common:: { task , watch , Pin , Poll } ;
18
+ use crate :: common:: Future ;
21
19
#[ cfg( any( feature = "http1" , feature = "http2" ) ) ]
22
20
#[ cfg( feature = "client" ) ]
23
- use crate :: common:: { Future , Never } ;
21
+ use crate :: common:: Never ;
22
+ use crate :: common:: { task, watch, Pin , Poll } ;
24
23
#[ cfg( all( feature = "http2" , any( feature = "client" , feature = "server" ) ) ) ]
25
24
use crate :: proto:: h2:: ping;
26
25
27
26
type BodySender = mpsc:: Sender < Result < Bytes , crate :: Error > > ;
27
+ type TrailersSender = oneshot:: Sender < HeaderMap > ;
28
28
29
29
/// A stream of `Bytes`, used when receiving bodies.
30
30
///
@@ -43,7 +43,8 @@ enum Kind {
43
43
Chan {
44
44
content_length : DecodedLength ,
45
45
want_tx : watch:: Sender ,
46
- rx : mpsc:: Receiver < Result < Bytes , crate :: Error > > ,
46
+ data_rx : mpsc:: Receiver < Result < Bytes , crate :: Error > > ,
47
+ trailers_rx : oneshot:: Receiver < HeaderMap > ,
47
48
} ,
48
49
#[ cfg( all( feature = "http2" , any( feature = "client" , feature = "server" ) ) ) ]
49
50
H2 {
@@ -106,7 +107,8 @@ enum DelayEof {
106
107
#[ must_use = "Sender does nothing unless sent on" ]
107
108
pub struct Sender {
108
109
want_rx : watch:: Receiver ,
109
- tx : BodySender ,
110
+ data_tx : BodySender ,
111
+ trailers_tx : Option < TrailersSender > ,
110
112
}
111
113
112
114
const WANT_PENDING : usize = 1 ;
@@ -137,19 +139,25 @@ impl Body {
137
139
}
138
140
139
141
pub ( crate ) fn new_channel ( content_length : DecodedLength , wanter : bool ) -> ( Sender , Body ) {
140
- let ( tx, rx) = mpsc:: channel ( 0 ) ;
142
+ let ( data_tx, data_rx) = mpsc:: channel ( 0 ) ;
143
+ let ( trailers_tx, trailers_rx) = oneshot:: channel ( ) ;
141
144
142
145
// If wanter is true, `Sender::poll_ready()` won't becoming ready
143
146
// until the `Body` has been polled for data once.
144
147
let want = if wanter { WANT_PENDING } else { WANT_READY } ;
145
148
146
149
let ( want_tx, want_rx) = watch:: channel ( want) ;
147
150
148
- let tx = Sender { want_rx, tx } ;
151
+ let tx = Sender {
152
+ want_rx,
153
+ data_tx,
154
+ trailers_tx : Some ( trailers_tx) ,
155
+ } ;
149
156
let rx = Body :: new ( Kind :: Chan {
150
157
content_length,
151
158
want_tx,
152
- rx,
159
+ data_rx,
160
+ trailers_rx,
153
161
} ) ;
154
162
155
163
( tx, rx)
@@ -282,12 +290,13 @@ impl Body {
282
290
Kind :: Once ( ref mut val) => Poll :: Ready ( val. take ( ) . map ( Ok ) ) ,
283
291
Kind :: Chan {
284
292
content_length : ref mut len,
285
- ref mut rx ,
293
+ ref mut data_rx ,
286
294
ref mut want_tx,
295
+ ..
287
296
} => {
288
297
want_tx. send ( WANT_READY ) ;
289
298
290
- match ready ! ( Pin :: new( rx ) . poll_next( cx) ?) {
299
+ match ready ! ( Pin :: new( data_rx ) . poll_next( cx) ?) {
291
300
Some ( chunk) => {
292
301
len. sub_if ( chunk. len ( ) as u64 ) ;
293
302
Poll :: Ready ( Some ( Ok ( chunk) ) )
@@ -368,10 +377,15 @@ impl HttpBody for Body {
368
377
}
369
378
Err ( e) => Poll :: Ready ( Err ( crate :: Error :: new_h2 ( e) ) ) ,
370
379
} ,
371
-
380
+ Kind :: Chan {
381
+ ref mut trailers_rx,
382
+ ..
383
+ } => match ready ! ( Pin :: new( trailers_rx) . poll( cx) ) {
384
+ Ok ( t) => Poll :: Ready ( Ok ( Some ( t) ) ) ,
385
+ Err ( _) => Poll :: Ready ( Ok ( None ) ) ,
386
+ } ,
372
387
#[ cfg( feature = "ffi" ) ]
373
388
Kind :: Ffi ( ref mut body) => body. poll_trailers ( cx) ,
374
-
375
389
_ => Poll :: Ready ( Ok ( None ) ) ,
376
390
}
377
391
}
@@ -527,7 +541,7 @@ impl Sender {
527
541
pub fn poll_ready ( & mut self , cx : & mut task:: Context < ' _ > ) -> Poll < crate :: Result < ( ) > > {
528
542
// Check if the receiver end has tried polling for the body yet
529
543
ready ! ( self . poll_want( cx) ?) ;
530
- self . tx
544
+ self . data_tx
531
545
. poll_ready ( cx)
532
546
. map_err ( |_| crate :: Error :: new_closed ( ) )
533
547
}
@@ -545,14 +559,23 @@ impl Sender {
545
559
futures_util:: future:: poll_fn ( |cx| self . poll_ready ( cx) ) . await
546
560
}
547
561
548
- /// Send data on this channel when it is ready.
562
+ /// Send data on data channel when it is ready.
549
563
pub async fn send_data ( & mut self , chunk : Bytes ) -> crate :: Result < ( ) > {
550
564
self . ready ( ) . await ?;
551
- self . tx
565
+ self . data_tx
552
566
. try_send ( Ok ( chunk) )
553
567
. map_err ( |_| crate :: Error :: new_closed ( ) )
554
568
}
555
569
570
+ /// Send trailers on trailers channel.
571
+ pub async fn send_trailers ( & mut self , trailers : HeaderMap ) -> crate :: Result < ( ) > {
572
+ let tx = match self . trailers_tx . take ( ) {
573
+ Some ( tx) => tx,
574
+ None => return Err ( crate :: Error :: new_closed ( ) ) ,
575
+ } ;
576
+ tx. send ( trailers) . map_err ( |_| crate :: Error :: new_closed ( ) )
577
+ }
578
+
556
579
/// Try to send data on this channel.
557
580
///
558
581
/// # Errors
@@ -566,23 +589,23 @@ impl Sender {
566
589
/// that doesn't have an async context. If in an async context, prefer
567
590
/// `send_data()` instead.
568
591
pub fn try_send_data ( & mut self , chunk : Bytes ) -> Result < ( ) , Bytes > {
569
- self . tx
592
+ self . data_tx
570
593
. try_send ( Ok ( chunk) )
571
594
. map_err ( |err| err. into_inner ( ) . expect ( "just sent Ok" ) )
572
595
}
573
596
574
597
/// Aborts the body in an abnormal fashion.
575
598
pub fn abort ( self ) {
576
599
let _ = self
577
- . tx
600
+ . data_tx
578
601
// clone so the send works even if buffer is full
579
602
. clone ( )
580
603
. try_send ( Err ( crate :: Error :: new_body_write_aborted ( ) ) ) ;
581
604
}
582
605
583
606
#[ cfg( feature = "http1" ) ]
584
607
pub ( crate ) fn send_error ( & mut self , err : crate :: Error ) {
585
- let _ = self . tx . try_send ( Err ( err) ) ;
608
+ let _ = self . data_tx . try_send ( Err ( err) ) ;
586
609
}
587
610
}
588
611
@@ -628,7 +651,7 @@ mod tests {
628
651
629
652
assert_eq ! (
630
653
mem:: size_of:: <Sender >( ) ,
631
- mem:: size_of:: <usize >( ) * 4 ,
654
+ mem:: size_of:: <usize >( ) * 5 ,
632
655
"Sender"
633
656
) ;
634
657
0 commit comments