reset pending push promises if user drops all refs (#199)
This commit is contained in:
@@ -87,6 +87,7 @@ impl Prioritize {
|
|||||||
pub fn schedule_send(&mut self, stream: &mut store::Ptr, task: &mut Option<Task>) {
|
pub fn schedule_send(&mut self, stream: &mut store::Ptr, task: &mut Option<Task>) {
|
||||||
// If the stream is waiting to be opened, nothing more to do.
|
// If the stream is waiting to be opened, nothing more to do.
|
||||||
if !stream.is_pending_open {
|
if !stream.is_pending_open {
|
||||||
|
trace!("schedule_send; {:?}", stream.id);
|
||||||
// Queue the stream
|
// Queue the stream
|
||||||
self.pending_send.push(stream);
|
self.pending_send.push(stream);
|
||||||
|
|
||||||
|
|||||||
@@ -615,6 +615,8 @@ impl Recv {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
trace!("enqueue_reset_expiration; {:?}", stream.id);
|
||||||
|
|
||||||
if !counts.can_inc_num_reset_streams() {
|
if !counts.can_inc_num_reset_streams() {
|
||||||
// try to evict 1 stream if possible
|
// try to evict 1 stream if possible
|
||||||
// if max allow is 0, this won't be able to evict,
|
// if max allow is 0, this won't be able to evict,
|
||||||
|
|||||||
@@ -909,16 +909,29 @@ fn drop_stream_ref(inner: &Mutex<Inner>, key: store::Key) {
|
|||||||
|
|
||||||
let actions = &mut me.actions;
|
let actions = &mut me.actions;
|
||||||
|
|
||||||
me.counts.transition(stream, |counts, mut stream| {
|
me.counts.transition(stream, |counts, stream| {
|
||||||
if stream.is_canceled_interest() {
|
maybe_cancel(stream, actions, counts);
|
||||||
actions.send.schedule_cancel(
|
|
||||||
&mut stream,
|
if stream.ref_count == 0 {
|
||||||
&mut actions.task);
|
let mut ppp = stream.pending_push_promises.take();
|
||||||
actions.recv.enqueue_reset_expiration(stream, counts);
|
while let Some(promise) = ppp.pop(stream.store_mut()) {
|
||||||
|
counts.transition(promise, |counts, stream| {
|
||||||
|
maybe_cancel(stream, actions, counts);
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn maybe_cancel(stream: &mut store::Ptr, actions: &mut Actions, counts: &mut Counts) {
|
||||||
|
if stream.is_canceled_interest() {
|
||||||
|
actions.send.schedule_cancel(
|
||||||
|
stream,
|
||||||
|
&mut actions.task);
|
||||||
|
actions.recv.enqueue_reset_expiration(stream, counts);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// ===== impl SendBuffer =====
|
// ===== impl SendBuffer =====
|
||||||
|
|
||||||
impl<B> SendBuffer<B> {
|
impl<B> SendBuffer<B> {
|
||||||
|
|||||||
@@ -94,6 +94,49 @@ fn recv_push_when_push_disabled_is_conn_error() {
|
|||||||
h2.join(mock).wait().unwrap();
|
h2.join(mock).wait().unwrap();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn pending_push_promises_reset_when_dropped() {
|
||||||
|
let _ = ::env_logger::init();
|
||||||
|
|
||||||
|
let (io, srv) = mock::new();
|
||||||
|
let srv = srv.assert_client_handshake()
|
||||||
|
.unwrap()
|
||||||
|
.recv_settings()
|
||||||
|
.recv_frame(
|
||||||
|
frames::headers(1)
|
||||||
|
.request("GET", "https://http2.akamai.com/")
|
||||||
|
.eos(),
|
||||||
|
)
|
||||||
|
.send_frame(
|
||||||
|
frames::push_promise(1, 2)
|
||||||
|
.request("GET", "https://http2.akamai.com/style.css")
|
||||||
|
)
|
||||||
|
.send_frame(frames::headers(1).response(200).eos())
|
||||||
|
.recv_frame(frames::reset(2).cancel())
|
||||||
|
.close();
|
||||||
|
|
||||||
|
let client = Client::handshake(io).unwrap().and_then(|(mut client, conn)| {
|
||||||
|
let request = Request::builder()
|
||||||
|
.method(Method::GET)
|
||||||
|
.uri("https://http2.akamai.com/")
|
||||||
|
.body(())
|
||||||
|
.unwrap();
|
||||||
|
let req = client
|
||||||
|
.send_request(request, true)
|
||||||
|
.unwrap()
|
||||||
|
.0.expect("response")
|
||||||
|
.and_then(|resp| {
|
||||||
|
assert_eq!(resp.status(), StatusCode::OK);
|
||||||
|
Ok(())
|
||||||
|
});
|
||||||
|
|
||||||
|
conn.drive(req)
|
||||||
|
.and_then(|(conn, _)| conn.expect("client"))
|
||||||
|
});
|
||||||
|
|
||||||
|
client.join(srv).wait().expect("wait");
|
||||||
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
#[ignore]
|
#[ignore]
|
||||||
fn recv_push_promise_with_unsafe_method_is_stream_error() {
|
fn recv_push_promise_with_unsafe_method_is_stream_error() {
|
||||||
|
|||||||
Reference in New Issue
Block a user