resets too
This commit is contained in:
22
src/error.rs
22
src/error.rs
@@ -22,7 +22,23 @@ pub enum ConnectionError {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub struct Stream(Reason);
|
pub struct StreamError(Reason);
|
||||||
|
|
||||||
|
impl StreamError {
|
||||||
|
pub fn new(r: Reason) -> StreamError {
|
||||||
|
StreamError(r)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn reason(&self) -> Reason {
|
||||||
|
self.0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<Reason> for StreamError {
|
||||||
|
fn from(r: Reason) -> Self {
|
||||||
|
StreamError(r)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#[derive(Debug, PartialEq, Eq, Clone, Copy)]
|
#[derive(Debug, PartialEq, Eq, Clone, Copy)]
|
||||||
pub enum Reason {
|
pub enum Reason {
|
||||||
@@ -65,6 +81,9 @@ pub enum User {
|
|||||||
/// The connection state is corrupt and the connection should be dropped.
|
/// The connection state is corrupt and the connection should be dropped.
|
||||||
Corrupt,
|
Corrupt,
|
||||||
|
|
||||||
|
/// The stream state has been reset.
|
||||||
|
StreamReset,
|
||||||
|
|
||||||
// TODO: reserve additional variants
|
// TODO: reserve additional variants
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -103,6 +122,7 @@ macro_rules! user_desc {
|
|||||||
InactiveStreamId => concat!($prefix, "inactive stream ID"),
|
InactiveStreamId => concat!($prefix, "inactive stream ID"),
|
||||||
UnexpectedFrameType => concat!($prefix, "unexpected frame type"),
|
UnexpectedFrameType => concat!($prefix, "unexpected frame type"),
|
||||||
FlowControlViolation => concat!($prefix, "flow control violation"),
|
FlowControlViolation => concat!($prefix, "flow control violation"),
|
||||||
|
StreamReset => concat!($prefix, "frame sent on reset stream"),
|
||||||
Corrupt => concat!($prefix, "connection state corrupt"),
|
Corrupt => concat!($prefix, "connection state corrupt"),
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -4,7 +4,6 @@ use bytes::{BufMut, Bytes, Buf};
|
|||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub struct Data<T = Bytes> {
|
pub struct Data<T = Bytes> {
|
||||||
stream_id: StreamId,
|
stream_id: StreamId,
|
||||||
//data_len: FrameSize,
|
|
||||||
data: T,
|
data: T,
|
||||||
flags: DataFlag,
|
flags: DataFlag,
|
||||||
pad_len: Option<u8>,
|
pad_len: Option<u8>,
|
||||||
@@ -29,7 +28,6 @@ impl Data<Bytes> {
|
|||||||
};
|
};
|
||||||
Ok(Data {
|
Ok(Data {
|
||||||
stream_id: head.stream_id(),
|
stream_id: head.stream_id(),
|
||||||
//data_len: payload.len() as FrameSize,
|
|
||||||
data: payload,
|
data: payload,
|
||||||
flags: flags,
|
flags: flags,
|
||||||
pad_len: pad_len,
|
pad_len: pad_len,
|
||||||
@@ -71,7 +69,6 @@ impl<T: Buf> Data<T> {
|
|||||||
}
|
}
|
||||||
Data {
|
Data {
|
||||||
stream_id,
|
stream_id,
|
||||||
//data_len: data.remaining() as FrameSize,
|
|
||||||
data,
|
data,
|
||||||
flags,
|
flags,
|
||||||
pad_len: None,
|
pad_len: None,
|
||||||
|
|||||||
@@ -48,6 +48,12 @@ pub struct PushPromise {
|
|||||||
flags: HeadersFlag,
|
flags: HeadersFlag,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl PushPromise {
|
||||||
|
pub fn stream_id(&self) -> StreamId {
|
||||||
|
self.stream_id
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub struct Continuation {
|
pub struct Continuation {
|
||||||
/// Stream ID of continuation frame
|
/// Stream ID of continuation frame
|
||||||
|
|||||||
@@ -61,7 +61,25 @@ pub enum Frame<T = Bytes> {
|
|||||||
PushPromise(PushPromise),
|
PushPromise(PushPromise),
|
||||||
Settings(Settings),
|
Settings(Settings),
|
||||||
Ping(Ping),
|
Ping(Ping),
|
||||||
WindowUpdate(WindowUpdate)
|
WindowUpdate(WindowUpdate),
|
||||||
|
Reset(Reset)
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T> Frame<T> {
|
||||||
|
pub fn stream_id(&self) -> StreamId {
|
||||||
|
use self::Frame::*;
|
||||||
|
|
||||||
|
match self {
|
||||||
|
&Headers(ref v) => v.stream_id(),
|
||||||
|
&Data(ref v) => v.stream_id(),
|
||||||
|
&PushPromise(ref v) => v.stream_id(),
|
||||||
|
&WindowUpdate(ref v) => v.stream_id(),
|
||||||
|
&Reset(ref v) => v.stream_id(),
|
||||||
|
|
||||||
|
&Ping(_) |
|
||||||
|
&Settings(_) => StreamId::zero(),
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Errors that can occur during parsing an HTTP/2 frame.
|
/// Errors that can occur during parsing an HTTP/2 frame.
|
||||||
|
|||||||
@@ -1,5 +1,7 @@
|
|||||||
use frame::{Head, Error};
|
use error::Reason;
|
||||||
use super::{StreamId};
|
use frame::{self, Head, Error, Kind, StreamId};
|
||||||
|
|
||||||
|
use bytes::{BufMut, BigEndian};
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub struct Reset {
|
pub struct Reset {
|
||||||
@@ -8,11 +10,15 @@ pub struct Reset {
|
|||||||
}
|
}
|
||||||
|
|
||||||
impl Reset {
|
impl Reset {
|
||||||
|
pub fn new(stream_id: StreamId, error: Reason) -> Reset {
|
||||||
|
Reset {
|
||||||
|
stream_id,
|
||||||
|
error_code: error.into(),
|
||||||
|
}
|
||||||
|
}
|
||||||
pub fn load(head: Head, payload: &[u8]) -> Result<Reset, Error> {
|
pub fn load(head: Head, payload: &[u8]) -> Result<Reset, Error> {
|
||||||
if payload.len() != 4 {
|
if payload.len() != 4 {
|
||||||
// Invalid payload len
|
return Err(Error::InvalidPayloadLength);
|
||||||
// TODO: Handle error
|
|
||||||
unimplemented!();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
let error_code = unpack_octets_4!(payload, 0, u32);
|
let error_code = unpack_octets_4!(payload, 0, u32);
|
||||||
@@ -22,4 +28,25 @@ impl Reset {
|
|||||||
error_code: error_code,
|
error_code: error_code,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn encode<B: BufMut>(&self, dst: &mut B) {
|
||||||
|
trace!("encoding RESET; id={:?} code={}", self.stream_id, self.error_code);
|
||||||
|
let head = Head::new(Kind::Reset, 0, self.stream_id);
|
||||||
|
head.encode(4, dst);
|
||||||
|
dst.put_u32::<BigEndian>(self.error_code);
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn stream_id(&self) -> StreamId {
|
||||||
|
self.stream_id
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn reason(&self) -> Reason {
|
||||||
|
self.error_code.into()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<B> From<Reset> for frame::Frame<B> {
|
||||||
|
fn from(src: Reset) -> Self {
|
||||||
|
frame::Frame::Reset(src)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
use StreamId;
|
use StreamId;
|
||||||
use byteorder::NetworkEndian;
|
|
||||||
use bytes::{BufMut};
|
|
||||||
use frame::{self, Head, Kind, Error};
|
use frame::{self, Head, Kind, Error};
|
||||||
|
|
||||||
|
use bytes::{BufMut, BigEndian};
|
||||||
|
|
||||||
const SIZE_INCREMENT_MASK: u32 = 1 << 31;
|
const SIZE_INCREMENT_MASK: u32 = 1 << 31;
|
||||||
|
|
||||||
#[derive(Copy, Clone, Debug)]
|
#[derive(Copy, Clone, Debug)]
|
||||||
@@ -48,7 +48,7 @@ impl WindowUpdate {
|
|||||||
trace!("encoding WINDOW_UPDATE; id={:?}", self.stream_id);
|
trace!("encoding WINDOW_UPDATE; id={:?}", self.stream_id);
|
||||||
let head = Head::new(Kind::Ping, 0, self.stream_id);
|
let head = Head::new(Kind::Ping, 0, self.stream_id);
|
||||||
head.encode(4, dst);
|
head.encode(4, dst);
|
||||||
dst.put_u32::<NetworkEndian>(self.size_increment);
|
dst.put_u32::<BigEndian>(self.size_increment);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -515,7 +515,7 @@ impl Table {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
fn assert_valid_state(&self, msg: &'static str) -> bool {
|
fn assert_valid_state(&self, _msg: &'static str) -> bool {
|
||||||
/*
|
/*
|
||||||
// Checks that the internal map structure is valid
|
// Checks that the internal map structure is valid
|
||||||
//
|
//
|
||||||
|
|||||||
@@ -34,7 +34,7 @@ pub mod server;
|
|||||||
|
|
||||||
mod util;
|
mod util;
|
||||||
|
|
||||||
pub use error::ConnectionError;
|
pub use error::{ConnectionError, Reason};
|
||||||
pub use frame::{StreamId};
|
pub use frame::{StreamId};
|
||||||
pub use proto::Connection;
|
pub use proto::Connection;
|
||||||
|
|
||||||
@@ -63,9 +63,9 @@ pub enum Frame<T, B = Bytes> {
|
|||||||
id: StreamId,
|
id: StreamId,
|
||||||
promise: (),
|
promise: (),
|
||||||
},
|
},
|
||||||
Error {
|
Reset {
|
||||||
id: StreamId,
|
id: StreamId,
|
||||||
error: (),
|
error: Reason,
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -2,25 +2,24 @@ use {ConnectionError, Frame};
|
|||||||
use client::Client;
|
use client::Client;
|
||||||
use error;
|
use error;
|
||||||
use frame::{self, SettingSet, StreamId};
|
use frame::{self, SettingSet, StreamId};
|
||||||
use proto::{self, ControlFlow, ControlPing, ControlSettings, Peer, PingPayload, ReadySink, WindowSize};
|
use proto::*;
|
||||||
use server::Server;
|
use server::Server;
|
||||||
|
|
||||||
use bytes::{Bytes, IntoBuf};
|
use bytes::{Bytes, IntoBuf};
|
||||||
use http::{request, response};
|
use http::{request, response};
|
||||||
use futures::*;
|
|
||||||
use tokio_io::{AsyncRead, AsyncWrite};
|
use tokio_io::{AsyncRead, AsyncWrite};
|
||||||
use std::marker::PhantomData;
|
use std::marker::PhantomData;
|
||||||
|
|
||||||
/// An H2 connection
|
/// An H2 connection
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub struct Connection<T, P, B: IntoBuf = Bytes> {
|
pub struct Connection<T, P, B: IntoBuf = Bytes> {
|
||||||
inner: proto::Transport<T, P, B::Buf>,
|
inner: Transport<T, P, B::Buf>,
|
||||||
// Set to `true` as long as the connection is in a valid state.
|
// Set to `true` as long as the connection is in a valid state.
|
||||||
active: bool,
|
active: bool,
|
||||||
_phantom: PhantomData<(P, B)>,
|
_phantom: PhantomData<(P, B)>,
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn new<T, P, B>(transport: proto::Transport<T, P, B::Buf>)
|
pub fn new<T, P, B>(transport: Transport<T, P, B::Buf>)
|
||||||
-> Connection<T, P, B>
|
-> Connection<T, P, B>
|
||||||
where T: AsyncRead + AsyncWrite,
|
where T: AsyncRead + AsyncWrite,
|
||||||
P: Peer,
|
P: Peer,
|
||||||
@@ -178,6 +177,11 @@ impl<T, P, B> Stream for Connection<T, P, B>
|
|||||||
data: v.into_payload(),
|
data: v.into_payload(),
|
||||||
},
|
},
|
||||||
|
|
||||||
|
Some(Reset(v)) => Frame::Reset {
|
||||||
|
id: v.stream_id(),
|
||||||
|
error: v.reason(),
|
||||||
|
},
|
||||||
|
|
||||||
Some(frame) => panic!("unexpected frame; frame={:?}", frame),
|
Some(frame) => panic!("unexpected frame; frame={:?}", frame),
|
||||||
None => return Ok(Async::Ready(None)),
|
None => return Ok(Async::Ready(None)),
|
||||||
};
|
};
|
||||||
@@ -213,6 +217,10 @@ impl<T, P, B> Sink for Connection<T, P, B>
|
|||||||
|
|
||||||
match item {
|
match item {
|
||||||
Frame::Headers { id, headers, end_of_stream } => {
|
Frame::Headers { id, headers, end_of_stream } => {
|
||||||
|
if self.inner.stream_is_reset(id) {
|
||||||
|
return Err(error::User::StreamReset.into());
|
||||||
|
}
|
||||||
|
|
||||||
// This is a one-way conversion. By checking `poll_ready` first (above),
|
// This is a one-way conversion. By checking `poll_ready` first (above),
|
||||||
// it's already been determined that the inner `Sink` can accept the item.
|
// it's already been determined that the inner `Sink` can accept the item.
|
||||||
// If the item is rejected, then there is a bug.
|
// If the item is rejected, then there is a bug.
|
||||||
@@ -222,20 +230,28 @@ impl<T, P, B> Sink for Connection<T, P, B>
|
|||||||
Ok(AsyncSink::Ready)
|
Ok(AsyncSink::Ready)
|
||||||
}
|
}
|
||||||
|
|
||||||
Frame::Data { id, data, end_of_stream, .. } => {
|
Frame::Data { id, data, end_of_stream } => {
|
||||||
|
if self.inner.stream_is_reset(id) {
|
||||||
|
return Err(error::User::StreamReset.into());
|
||||||
|
}
|
||||||
|
|
||||||
let frame = frame::Data::from_buf(id, data.into_buf(), end_of_stream);
|
let frame = frame::Data::from_buf(id, data.into_buf(), end_of_stream);
|
||||||
let res = try!(self.inner.start_send(frame.into()));
|
let res = try!(self.inner.start_send(frame.into()));
|
||||||
assert!(res.is_ready());
|
assert!(res.is_ready());
|
||||||
Ok(AsyncSink::Ready)
|
Ok(AsyncSink::Ready)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Frame::Reset { id, error } => {
|
||||||
|
let f = frame::Reset::new(id, error);
|
||||||
|
let res = self.inner.start_send(f.into())?;
|
||||||
|
assert!(res.is_ready());
|
||||||
|
Ok(AsyncSink::Ready)
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Frame::Trailers { id, headers } => {
|
Frame::Trailers { id, headers } => {
|
||||||
unimplemented!();
|
unimplemented!();
|
||||||
}
|
}
|
||||||
Frame::Body { id, chunk, end_of_stream } => {
|
|
||||||
unimplemented!();
|
|
||||||
}
|
|
||||||
Frame::PushPromise { id, promise } => {
|
Frame::PushPromise { id, promise } => {
|
||||||
unimplemented!();
|
unimplemented!();
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -76,26 +76,33 @@ impl<T: ControlStreams> FlowControl<T> {
|
|||||||
|
|
||||||
/// Proxies access to streams.
|
/// Proxies access to streams.
|
||||||
impl<T: ControlStreams> ControlStreams for FlowControl<T> {
|
impl<T: ControlStreams> ControlStreams for FlowControl<T> {
|
||||||
#[inline]
|
|
||||||
fn streams(&self) -> &StreamMap {
|
fn streams(&self) -> &StreamMap {
|
||||||
self.inner.streams()
|
self.inner.streams()
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
|
||||||
fn streams_mut(&mut self) -> &mut StreamMap {
|
fn streams_mut(&mut self) -> &mut StreamMap {
|
||||||
self.inner.streams_mut()
|
self.inner.streams_mut()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn stream_is_reset(&self, id: StreamId) -> bool {
|
||||||
|
self.inner.stream_is_reset(id)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Exposes a public upward API for flow control.
|
/// Exposes a public upward API for flow control.
|
||||||
impl<T: ControlStreams> ControlFlow for FlowControl<T> {
|
impl<T: ControlStreams> ControlFlow for FlowControl<T> {
|
||||||
fn poll_remote_window_update(&mut self, id: StreamId) -> Poll<WindowSize, ConnectionError> {
|
fn poll_remote_window_update(&mut self, id: StreamId) -> Poll<WindowSize, ConnectionError> {
|
||||||
if let Some(mut flow) = self.remote_flow_controller(id) {
|
if self.stream_is_reset(id) {
|
||||||
if let Some(sz) = flow.apply_window_update() {
|
return Err(error::User::StreamReset.into());
|
||||||
return Ok(Async::Ready(sz));
|
}
|
||||||
|
|
||||||
|
match self.remote_flow_controller(id) {
|
||||||
|
None => return Err(error::User::InvalidStreamId.into()),
|
||||||
|
Some(mut flow) => {
|
||||||
|
if let Some(sz) = flow.apply_window_update() {
|
||||||
|
return Ok(Async::Ready(sz));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
return Err(error::User::InvalidStreamId.into());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
self.blocked_remote_window_update = Some(task::current());
|
self.blocked_remote_window_update = Some(task::current());
|
||||||
@@ -103,18 +110,26 @@ impl<T: ControlStreams> ControlFlow for FlowControl<T> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fn expand_local_window(&mut self, id: StreamId, incr: WindowSize) -> Result<(), ConnectionError> {
|
fn expand_local_window(&mut self, id: StreamId, incr: WindowSize) -> Result<(), ConnectionError> {
|
||||||
if let Some(mut fc) = self.local_flow_controller(id) {
|
let added = match self.local_flow_controller(id) {
|
||||||
fc.expand_window(incr);
|
None => false,
|
||||||
} else {
|
Some(mut fc) => {
|
||||||
return Err(error::User::InvalidStreamId.into());
|
fc.expand_window(incr);
|
||||||
}
|
true
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
if id.is_zero() {
|
if added {
|
||||||
self.pending_local_connection_window_update = true;
|
if id.is_zero() {
|
||||||
|
self.pending_local_connection_window_update = true;
|
||||||
|
} else {
|
||||||
|
self.pending_local_window_updates.push_back(id);
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
|
} else if self.stream_is_reset(id) {
|
||||||
|
Err(error::User::StreamReset.into())
|
||||||
} else {
|
} else {
|
||||||
self.pending_local_window_updates.push_back(id);
|
Err(error::User::InvalidStreamId.into())
|
||||||
}
|
}
|
||||||
Ok(())
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -145,9 +160,11 @@ impl<T, U> FlowControl<T>
|
|||||||
}
|
}
|
||||||
|
|
||||||
while let Some(id) = self.pending_local_window_updates.pop_front() {
|
while let Some(id) = self.pending_local_window_updates.pop_front() {
|
||||||
let update = self.local_flow_controller(id).and_then(|s| s.apply_window_update());
|
if !self.stream_is_reset(id) {
|
||||||
if let Some(incr) = update {
|
let update = self.local_flow_controller(id).and_then(|s| s.apply_window_update());
|
||||||
try_ready!(self.try_send(frame::WindowUpdate::new(id, incr)));
|
if let Some(incr) = update {
|
||||||
|
try_ready!(self.try_send(frame::WindowUpdate::new(id, incr)));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -80,12 +80,12 @@ impl<T> FramedRead<T> {
|
|||||||
frame.into()
|
frame.into()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Kind::Reset => {
|
||||||
|
frame::Reset::load(head, &bytes[frame::HEADER_LEN..])?.into()
|
||||||
|
}
|
||||||
|
|
||||||
// TODO
|
// TODO
|
||||||
|
|
||||||
Kind::Reset => {
|
|
||||||
let _todo = try!(frame::Reset::load(head, &bytes[frame::HEADER_LEN..]));
|
|
||||||
unimplemented!();
|
|
||||||
}
|
|
||||||
Kind::GoAway => {
|
Kind::GoAway => {
|
||||||
let _todo = try!(frame::GoAway::load(&bytes[frame::HEADER_LEN..]));
|
let _todo = try!(frame::GoAway::load(&bytes[frame::HEADER_LEN..]));
|
||||||
unimplemented!();
|
unimplemented!();
|
||||||
|
|||||||
@@ -145,6 +145,10 @@ impl<T, B> Sink for FramedWrite<T, B>
|
|||||||
v.encode(self.buf.get_mut());
|
v.encode(self.buf.get_mut());
|
||||||
trace!("encoded window_update; rem={:?}", self.buf.remaining());
|
trace!("encoded window_update; rem={:?}", self.buf.remaining());
|
||||||
}
|
}
|
||||||
|
Frame::Reset(v) => {
|
||||||
|
v.encode(self.buf.get_mut());
|
||||||
|
trace!("encoded reset; rem={:?}", self.buf.remaining());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(AsyncSink::Ready)
|
Ok(AsyncSink::Ready)
|
||||||
|
|||||||
@@ -128,6 +128,7 @@ pub trait ControlFlow {
|
|||||||
pub trait ControlStreams {
|
pub trait ControlStreams {
|
||||||
fn streams(&self)-> &StreamMap;
|
fn streams(&self)-> &StreamMap;
|
||||||
fn streams_mut(&mut self) -> &mut StreamMap;
|
fn streams_mut(&mut self) -> &mut StreamMap;
|
||||||
|
fn stream_is_reset(&self, id: StreamId) -> bool;
|
||||||
}
|
}
|
||||||
|
|
||||||
pub type PingPayload = [u8; 8];
|
pub type PingPayload = [u8; 8];
|
||||||
|
|||||||
@@ -103,6 +103,10 @@ impl<T: ControlStreams> ControlStreams for Settings<T> {
|
|||||||
fn streams_mut(&mut self) -> &mut StreamMap {
|
fn streams_mut(&mut self) -> &mut StreamMap {
|
||||||
self.inner.streams_mut()
|
self.inner.streams_mut()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn stream_is_reset(&self, id: StreamId) -> bool {
|
||||||
|
self.inner.stream_is_reset(id)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<T: ControlFlow> ControlFlow for Settings<T> {
|
impl<T: ControlFlow> ControlFlow for Settings<T> {
|
||||||
|
|||||||
@@ -64,6 +64,14 @@ pub enum StreamState {
|
|||||||
}
|
}
|
||||||
|
|
||||||
impl StreamState {
|
impl StreamState {
|
||||||
|
pub fn is_closed(&self) -> bool {
|
||||||
|
use self::StreamState::*;
|
||||||
|
|
||||||
|
match self {
|
||||||
|
&Closed => true,
|
||||||
|
_ => false,
|
||||||
|
}
|
||||||
|
}
|
||||||
/// Transition the state to represent headers being received.
|
/// Transition the state to represent headers being received.
|
||||||
///
|
///
|
||||||
/// Returns true if this state transition results in iniitializing the
|
/// Returns true if this state transition results in iniitializing the
|
||||||
@@ -296,6 +304,10 @@ impl StreamMap {
|
|||||||
self.inner.entry(id)
|
self.inner.entry(id)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn remove(&mut self, id: &StreamId) -> Option<StreamState> {
|
||||||
|
self.inner.remove(id)
|
||||||
|
}
|
||||||
|
|
||||||
pub fn shrink_all_local_windows(&mut self, decr: u32) {
|
pub fn shrink_all_local_windows(&mut self, decr: u32) {
|
||||||
for (_, mut s) in &mut self.inner {
|
for (_, mut s) in &mut self.inner {
|
||||||
if let Some(fc) = s.local_flow_controller() {
|
if let Some(fc) = s.local_flow_controller() {
|
||||||
|
|||||||
@@ -1,16 +1,21 @@
|
|||||||
use ConnectionError;
|
use {ConnectionError};
|
||||||
use error::Reason;
|
use error::Reason;
|
||||||
use error::User;
|
use error::User;
|
||||||
use frame::{self, Frame};
|
use frame::{self, Frame};
|
||||||
use proto::*;
|
use proto::*;
|
||||||
|
|
||||||
|
use fnv::FnvHasher;
|
||||||
|
use ordermap::OrderMap;
|
||||||
|
use std::hash::BuildHasherDefault;
|
||||||
use std::marker::PhantomData;
|
use std::marker::PhantomData;
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub struct StreamTracker<T, P> {
|
pub struct StreamTracker<T, P> {
|
||||||
inner: T,
|
inner: T,
|
||||||
peer: PhantomData<P>,
|
peer: PhantomData<P>,
|
||||||
streams: StreamMap,
|
active_streams: StreamMap,
|
||||||
|
// TODO reserved_streams: HashSet<StreamId>
|
||||||
|
reset_streams: OrderMap<StreamId, Reason, BuildHasherDefault<FnvHasher>>,
|
||||||
local_max_concurrency: Option<u32>,
|
local_max_concurrency: Option<u32>,
|
||||||
remote_max_concurrency: Option<u32>,
|
remote_max_concurrency: Option<u32>,
|
||||||
initial_local_window_size: WindowSize,
|
initial_local_window_size: WindowSize,
|
||||||
@@ -32,7 +37,8 @@ impl<T, P, U> StreamTracker<T, P>
|
|||||||
StreamTracker {
|
StreamTracker {
|
||||||
inner,
|
inner,
|
||||||
peer: PhantomData,
|
peer: PhantomData,
|
||||||
streams: StreamMap::default(),
|
active_streams: StreamMap::default(),
|
||||||
|
reset_streams: OrderMap::default(),
|
||||||
local_max_concurrency,
|
local_max_concurrency,
|
||||||
remote_max_concurrency,
|
remote_max_concurrency,
|
||||||
initial_local_window_size,
|
initial_local_window_size,
|
||||||
@@ -44,12 +50,16 @@ impl<T, P, U> StreamTracker<T, P>
|
|||||||
impl<T, P> ControlStreams for StreamTracker<T, P> {
|
impl<T, P> ControlStreams for StreamTracker<T, P> {
|
||||||
#[inline]
|
#[inline]
|
||||||
fn streams(&self) -> &StreamMap {
|
fn streams(&self) -> &StreamMap {
|
||||||
&self.streams
|
&self.active_streams
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn streams_mut(&mut self) -> &mut StreamMap {
|
fn streams_mut(&mut self) -> &mut StreamMap {
|
||||||
&mut self.streams
|
&mut self.active_streams
|
||||||
|
}
|
||||||
|
|
||||||
|
fn stream_is_reset(&self, id: StreamId) -> bool {
|
||||||
|
self.reset_streams.contains_key(&id)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -109,40 +119,93 @@ impl<T, P> Stream for StreamTracker<T, P>
|
|||||||
fn poll(&mut self) -> Poll<Option<T::Item>, T::Error> {
|
fn poll(&mut self) -> Poll<Option<T::Item>, T::Error> {
|
||||||
use frame::Frame::*;
|
use frame::Frame::*;
|
||||||
|
|
||||||
match try_ready!(self.inner.poll()) {
|
loop {
|
||||||
Some(Headers(v)) => {
|
match try_ready!(self.inner.poll()) {
|
||||||
let id = v.stream_id();
|
Some(Headers(v)) => {
|
||||||
let eos = v.is_end_stream();
|
let id = v.stream_id();
|
||||||
|
let eos = v.is_end_stream();
|
||||||
|
|
||||||
let initialized = self.streams
|
if self.reset_streams.contains_key(&id) {
|
||||||
.entry(id)
|
continue;
|
||||||
.or_insert_with(|| StreamState::default())
|
}
|
||||||
.recv_headers::<P>(eos, self.initial_local_window_size)?;
|
|
||||||
|
|
||||||
if initialized {
|
let is_closed = {
|
||||||
// TODO: Ensure available capacity for a new stream
|
let stream = self.active_streams.entry(id)
|
||||||
// This won't be as simple as self.streams.len() as closed
|
.or_insert_with(|| StreamState::default());
|
||||||
// connections should not be factored.
|
|
||||||
|
|
||||||
if !P::is_valid_remote_stream_id(id) {
|
let initialized =
|
||||||
return Err(Reason::ProtocolError.into());
|
stream.recv_headers::<P>(eos, self.initial_local_window_size)?;
|
||||||
|
|
||||||
|
if initialized {
|
||||||
|
// TODO: Ensure available capacity for a new stream
|
||||||
|
// This won't be as simple as self.streams.len() as closed
|
||||||
|
// connections should not be factored.
|
||||||
|
|
||||||
|
if !P::is_valid_remote_stream_id(id) {
|
||||||
|
return Err(Reason::ProtocolError.into());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
stream.is_closed()
|
||||||
|
};
|
||||||
|
|
||||||
|
if is_closed {
|
||||||
|
self.active_streams.remove(&id);
|
||||||
|
self.reset_streams.insert(id, Reason::NoError);
|
||||||
|
}
|
||||||
|
|
||||||
|
return Ok(Async::Ready(Some(Headers(v))));
|
||||||
|
}
|
||||||
|
|
||||||
|
Some(Data(v)) => {
|
||||||
|
let id = v.stream_id();
|
||||||
|
|
||||||
|
if self.reset_streams.contains_key(&id) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
let is_closed = {
|
||||||
|
let stream = match self.active_streams.get_mut(&id) {
|
||||||
|
None => return Err(Reason::ProtocolError.into()),
|
||||||
|
Some(s) => s,
|
||||||
|
};
|
||||||
|
stream.recv_data(v.is_end_stream())?;
|
||||||
|
stream.is_closed()
|
||||||
|
};
|
||||||
|
|
||||||
|
if is_closed {
|
||||||
|
self.active_streams.remove(&id);
|
||||||
|
self.reset_streams.insert(id, Reason::NoError);
|
||||||
|
}
|
||||||
|
|
||||||
|
return Ok(Async::Ready(Some(Data(v))));
|
||||||
|
}
|
||||||
|
|
||||||
|
Some(Reset(v)) => {
|
||||||
|
let id = v.stream_id();
|
||||||
|
|
||||||
|
// Set or update the reset reason.
|
||||||
|
self.reset_streams.insert(id, v.reason());
|
||||||
|
|
||||||
|
if self.active_streams.remove(&id).is_some() {
|
||||||
|
return Ok(Async::Ready(Some(Reset(v))));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(Async::Ready(Some(Headers(v))))
|
Some(f) => {
|
||||||
}
|
let id = f.stream_id();
|
||||||
|
|
||||||
Some(Data(v)) => {
|
if self.reset_streams.contains_key(&id) {
|
||||||
match self.streams.get_mut(&v.stream_id()) {
|
continue;
|
||||||
None => Err(Reason::ProtocolError.into()),
|
|
||||||
Some(state) => {
|
|
||||||
state.recv_data(v.is_end_stream())?;
|
|
||||||
Ok(Async::Ready(Some(Data(v))))
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return Ok(Async::Ready(Some(f)));
|
||||||
|
}
|
||||||
|
|
||||||
|
None => {
|
||||||
|
return Ok(Async::Ready(None));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
f => Ok(Async::Ready(f)),
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -158,6 +221,9 @@ impl<T, P, U> Sink for StreamTracker<T, P>
|
|||||||
fn start_send(&mut self, item: T::SinkItem) -> StartSend<T::SinkItem, T::SinkError> {
|
fn start_send(&mut self, item: T::SinkItem) -> StartSend<T::SinkItem, T::SinkError> {
|
||||||
use frame::Frame::*;
|
use frame::Frame::*;
|
||||||
|
|
||||||
|
// Must be enforced through higher levels.
|
||||||
|
debug_assert!(!self.stream_is_reset(item.stream_id()));
|
||||||
|
|
||||||
match &item {
|
match &item {
|
||||||
&Headers(ref v) => {
|
&Headers(ref v) => {
|
||||||
let id = v.stream_id();
|
let id = v.stream_id();
|
||||||
@@ -170,34 +236,51 @@ impl<T, P, U> Sink for StreamTracker<T, P>
|
|||||||
//
|
//
|
||||||
// ACTUALLY(ver), maybe not?
|
// ACTUALLY(ver), maybe not?
|
||||||
// https://github.com/http2/http2-spec/commit/c83c8d911e6b6226269877e446a5cad8db921784
|
// https://github.com/http2/http2-spec/commit/c83c8d911e6b6226269877e446a5cad8db921784
|
||||||
let initialized = self.streams
|
|
||||||
.entry(id)
|
|
||||||
.or_insert_with(|| StreamState::default())
|
|
||||||
.send_headers::<P>(eos, self.initial_remote_window_size)?;
|
|
||||||
|
|
||||||
if initialized {
|
let is_closed = {
|
||||||
// TODO: Ensure available capacity for a new stream
|
let stream = self.active_streams.entry(id)
|
||||||
// This won't be as simple as self.streams.len() as closed
|
.or_insert_with(|| StreamState::default());
|
||||||
// connections should not be factored.
|
|
||||||
if !P::is_valid_local_stream_id(id) {
|
let initialized =
|
||||||
// TODO: clear state
|
stream.send_headers::<P>(eos, self.initial_remote_window_size)?;
|
||||||
return Err(User::InvalidStreamId.into());
|
|
||||||
|
if initialized {
|
||||||
|
// TODO: Ensure available capacity for a new stream
|
||||||
|
// This won't be as simple as self.streams.len() as closed
|
||||||
|
// connections should not be factored.
|
||||||
|
if !P::is_valid_local_stream_id(id) {
|
||||||
|
// TODO: clear state
|
||||||
|
return Err(User::InvalidStreamId.into());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
stream.is_closed()
|
||||||
|
};
|
||||||
|
|
||||||
|
if is_closed {
|
||||||
|
self.active_streams.remove(&id);
|
||||||
|
self.reset_streams.insert(id, Reason::NoError);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
&Data(ref v) => {
|
&Data(ref v) => {
|
||||||
match self.streams.get_mut(&v.stream_id()) {
|
match self.active_streams.get_mut(&v.stream_id()) {
|
||||||
None => return Err(User::InactiveStreamId.into()),
|
None => return Err(User::InactiveStreamId.into()),
|
||||||
Some(state) => state.send_data(v.is_end_stream())?,
|
Some(stream) => stream.send_data(v.is_end_stream())?,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
&Reset(ref v) => {
|
||||||
|
let id = v.stream_id();
|
||||||
|
self.active_streams.remove(&id);
|
||||||
|
self.reset_streams.insert(id, v.reason());
|
||||||
|
}
|
||||||
|
|
||||||
_ => {}
|
_ => {}
|
||||||
}
|
}
|
||||||
|
|
||||||
self.inner.start_send(item)
|
self.inner.start_send(item)
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fn poll_complete(&mut self) -> Poll<(), T::SinkError> {
|
fn poll_complete(&mut self) -> Poll<(), T::SinkError> {
|
||||||
|
|||||||
Reference in New Issue
Block a user