80
80
//! [`AsyncWrite`]: tokio::io::AsyncWrite
81
81
//! [`Connection`]: Connection
82
82
use std:: fmt;
83
+ use std:: fmt:: { Debug , Formatter } ;
84
+ use std:: sync:: atomic:: { AtomicBool , Ordering } ;
85
+ use std:: sync:: Arc ;
83
86
84
87
use :: http:: Extensions ;
85
88
@@ -113,6 +116,34 @@ pub struct Connected {
113
116
pub ( super ) alpn : Alpn ,
114
117
pub ( super ) is_proxied : bool ,
115
118
pub ( super ) extra : Option < Extra > ,
119
+ pub ( super ) poisoned : PoisonPill ,
120
+ }
121
+
122
+ #[ derive( Clone ) ]
123
+ pub ( crate ) struct PoisonPill {
124
+ poisoned : Arc < AtomicBool > ,
125
+ }
126
+
127
+ impl Debug for PoisonPill {
128
+ fn fmt ( & self , f : & mut Formatter < ' _ > ) -> fmt:: Result {
129
+ // print the address of the pill—this makes debugging issues much easier
130
+ write ! ( f, "PoisonPill@{:p} {{ poisoned: {} }}" , self . poisoned, self . poisoned. load( Ordering :: Relaxed ) )
131
+ }
132
+ }
133
+
134
+ impl PoisonPill {
135
+ pub ( crate ) fn healthy ( ) -> Self {
136
+ Self {
137
+ poisoned : Arc :: new ( AtomicBool :: new ( false ) ) ,
138
+ }
139
+ }
140
+ pub ( crate ) fn poison ( & self ) {
141
+ self . poisoned . store ( true , Ordering :: Relaxed )
142
+ }
143
+
144
+ pub ( crate ) fn poisoned ( & self ) -> bool {
145
+ self . poisoned . load ( Ordering :: Relaxed )
146
+ }
116
147
}
117
148
118
149
pub ( super ) struct Extra ( Box < dyn ExtraInner > ) ;
@@ -130,6 +161,7 @@ impl Connected {
130
161
alpn : Alpn :: None ,
131
162
is_proxied : false ,
132
163
extra : None ,
164
+ poisoned : PoisonPill :: healthy ( ) ,
133
165
}
134
166
}
135
167
@@ -189,6 +221,16 @@ impl Connected {
189
221
self . alpn == Alpn :: H2
190
222
}
191
223
224
+ /// Poison this connection
225
+ ///
226
+ /// A poisoned connection will not be reused for subsequent requests by the pool
227
+ pub fn poison ( & self ) {
228
+ self . poisoned . poison ( ) ;
229
+ tracing:: debug!(
230
+ poison_pill = ?self . poisoned, "connection was poisoned"
231
+ ) ;
232
+ }
233
+
192
234
// Don't public expose that `Connected` is `Clone`, unsure if we want to
193
235
// keep that contract...
194
236
#[ cfg( feature = "http2" ) ]
@@ -197,6 +239,7 @@ impl Connected {
197
239
alpn : self . alpn . clone ( ) ,
198
240
is_proxied : self . is_proxied ,
199
241
extra : self . extra . clone ( ) ,
242
+ poisoned : self . poisoned . clone ( ) ,
200
243
}
201
244
}
202
245
}
0 commit comments