cleanup codec::Framed

This commit is contained in:
Nikolay Kim 2020-03-31 10:01:06 +06:00
parent 20a404ed1e
commit a440ebb345
10 changed files with 93 additions and 126 deletions

View file

@ -17,9 +17,9 @@ path = "src/lib.rs"
[dependencies] [dependencies]
bitflags = "1.2.1" bitflags = "1.2.1"
bytes = "0.5.2" bytes = "0.5.4"
futures-core = "0.3.1" futures-core = "0.3.4"
futures-sink = "0.3.1" futures-sink = "0.3.4"
tokio = { version = "0.2.4", default-features=false } tokio = { version = "0.2.4", default-features=false }
tokio-util = { version = "0.2.0", default-features=false, features=["codec"] } tokio-util = { version = "0.2.0", default-features=false, features=["codec"] }
log = "0.4" log = "0.4"

View file

@ -195,16 +195,17 @@ impl<T, U> Framed<T, U> {
} }
} }
impl<T, U> Framed<T, U> { impl<T, U> Framed<T, U>
where
T: AsyncWrite + Unpin,
U: Encoder,
{
#[inline]
/// Serialize item and Write to the inner buffer /// Serialize item and Write to the inner buffer
pub fn write( pub fn write(
&mut self, &mut self,
item: <U as Encoder>::Item, item: <U as Encoder>::Item,
) -> Result<(), <U as Encoder>::Error> ) -> Result<(), <U as Encoder>::Error> {
where
T: AsyncWrite,
U: Encoder,
{
let remaining = self.write_buf.capacity() - self.write_buf.len(); let remaining = self.write_buf.capacity() - self.write_buf.len();
if remaining < LW { if remaining < LW {
self.write_buf.reserve(HW - remaining); self.write_buf.reserve(HW - remaining);
@ -214,6 +215,7 @@ impl<T, U> Framed<T, U> {
Ok(()) Ok(())
} }
#[inline]
/// Check if framed is able to write more data. /// Check if framed is able to write more data.
/// ///
/// `Framed` object considers ready if there is free space in write buffer. /// `Framed` object considers ready if there is free space in write buffer.
@ -221,6 +223,48 @@ impl<T, U> Framed<T, U> {
self.write_buf.len() < HW self.write_buf.len() < HW
} }
/// Flush write buffer to underlying I/O stream.
pub fn flush(&mut self, cx: &mut Context<'_>) -> Poll<Result<(), U::Error>> {
log::trace!("flushing framed transport");
while !self.write_buf.is_empty() {
log::trace!("writing; remaining={}", self.write_buf.len());
let n = ready!(Pin::new(&mut self.io).poll_write(cx, &self.write_buf))?;
if n == 0 {
return Poll::Ready(Err(io::Error::new(
io::ErrorKind::WriteZero,
"failed to write frame to transport",
)
.into()));
}
// remove written data
self.write_buf.advance(n);
}
// Try flushing the underlying IO
ready!(Pin::new(&mut self.io).poll_flush(cx))?;
log::trace!("framed transport flushed");
Poll::Ready(Ok(()))
}
#[inline]
/// Flush write buffer and shutdown underlying I/O stream.
pub fn close(&mut self, cx: &mut Context<'_>) -> Poll<Result<(), U::Error>> {
ready!(Pin::new(&mut self.io).poll_flush(cx))?;
ready!(Pin::new(&mut self.io).poll_shutdown(cx))?;
log::trace!("framed transport flushed and closed");
Poll::Ready(Ok(()))
}
}
impl<T, U> Framed<T, U>
where
T: AsyncRead + Unpin,
U: Decoder,
{
/// Try to read underlying I/O stream and decode item. /// Try to read underlying I/O stream and decode item.
pub fn next_item( pub fn next_item(
&mut self, &mut self,
@ -267,9 +311,8 @@ impl<T, U> Framed<T, U> {
if remaining < LW { if remaining < LW {
self.read_buf.reserve(HW - remaining) self.read_buf.reserve(HW - remaining)
} }
let cnt = match unsafe { let cnt = match Pin::new(&mut self.io).poll_read_buf(cx, &mut self.read_buf)
Pin::new_unchecked(&mut self.io).poll_read_buf(cx, &mut self.read_buf) {
} {
Poll::Pending => return Poll::Pending, Poll::Pending => return Poll::Pending,
Poll::Ready(Err(e)) => return Poll::Ready(Some(Err(e.into()))), Poll::Ready(Err(e)) => return Poll::Ready(Some(Err(e.into()))),
Poll::Ready(Ok(cnt)) => cnt, Poll::Ready(Ok(cnt)) => cnt,
@ -281,62 +324,16 @@ impl<T, U> Framed<T, U> {
self.flags.insert(Flags::READABLE); self.flags.insert(Flags::READABLE);
} }
} }
/// Flush write buffer to underlying I/O stream.
pub fn flush(&mut self, cx: &mut Context<'_>) -> Poll<Result<(), U::Error>>
where
T: AsyncWrite,
U: Encoder,
{
log::trace!("flushing framed transport");
while !self.write_buf.is_empty() {
log::trace!("writing; remaining={}", self.write_buf.len());
let n = ready!(unsafe {
Pin::new_unchecked(&mut self.io).poll_write(cx, &self.write_buf)
})?;
if n == 0 {
return Poll::Ready(Err(io::Error::new(
io::ErrorKind::WriteZero,
"failed to write frame to transport",
)
.into()));
}
// remove written data
self.write_buf.advance(n);
}
// Try flushing the underlying IO
ready!(unsafe { Pin::new_unchecked(&mut self.io).poll_flush(cx) })?;
log::trace!("framed transport flushed");
Poll::Ready(Ok(()))
}
/// Flush write buffer and shutdown underlying I/O stream.
pub fn close(&mut self, cx: &mut Context<'_>) -> Poll<Result<(), U::Error>>
where
T: AsyncWrite,
U: Encoder,
{
unsafe {
ready!(Pin::new_unchecked(&mut self.io).poll_flush(cx))?;
ready!(Pin::new_unchecked(&mut self.io).poll_shutdown(cx))?;
}
Poll::Ready(Ok(()))
}
} }
impl<T, U> Stream for Framed<T, U> impl<T, U> Stream for Framed<T, U>
where where
T: AsyncRead, T: AsyncRead + Unpin,
U: Decoder, U: Decoder,
{ {
type Item = Result<U::Item, U::Error>; type Item = Result<U::Item, U::Error>;
#[inline]
fn poll_next( fn poll_next(
mut self: Pin<&mut Self>, mut self: Pin<&mut Self>,
cx: &mut Context<'_>, cx: &mut Context<'_>,
@ -347,12 +344,13 @@ where
impl<T, U> Sink<U::Item> for Framed<T, U> impl<T, U> Sink<U::Item> for Framed<T, U>
where where
T: AsyncWrite, T: AsyncWrite + Unpin,
U: Encoder, U: Encoder,
U::Error: From<io::Error>, U::Error: From<io::Error>,
{ {
type Error = U::Error; type Error = U::Error;
#[inline]
fn poll_ready( fn poll_ready(
self: Pin<&mut Self>, self: Pin<&mut Self>,
_: &mut Context<'_>, _: &mut Context<'_>,
@ -364,6 +362,7 @@ where
} }
} }
#[inline]
fn start_send( fn start_send(
mut self: Pin<&mut Self>, mut self: Pin<&mut Self>,
item: <U as Encoder>::Item, item: <U as Encoder>::Item,
@ -371,6 +370,7 @@ where
self.write(item) self.write(item)
} }
#[inline]
fn poll_flush( fn poll_flush(
mut self: Pin<&mut Self>, mut self: Pin<&mut Self>,
cx: &mut Context<'_>, cx: &mut Context<'_>,
@ -378,6 +378,7 @@ where
self.flush(cx) self.flush(cx)
} }
#[inline]
fn poll_close( fn poll_close(
mut self: Pin<&mut Self>, mut self: Pin<&mut Self>,
cx: &mut Context<'_>, cx: &mut Context<'_>,

View file

@ -17,7 +17,7 @@ where
impl<Io, Codec> Connect<Io, Codec> impl<Io, Codec> Connect<Io, Codec>
where where
Io: AsyncRead + AsyncWrite, Io: AsyncRead + AsyncWrite + Unpin,
Codec: Encoder + Decoder, Codec: Encoder + Decoder,
{ {
pub(crate) fn new(io: Io) -> Self { pub(crate) fn new(io: Io) -> Self {
@ -80,7 +80,7 @@ impl<Io, St, Codec: Encoder + Decoder, Out: Unpin> ConnectResult<Io, St, Codec,
impl<Io, St, Codec, Out> Stream for ConnectResult<Io, St, Codec, Out> impl<Io, St, Codec, Out> Stream for ConnectResult<Io, St, Codec, Out>
where where
Io: AsyncRead + AsyncWrite, Io: AsyncRead + AsyncWrite + Unpin,
Codec: Encoder + Decoder, Codec: Encoder + Decoder,
{ {
type Item = Result<<Codec as Decoder>::Item, <Codec as Decoder>::Error>; type Item = Result<<Codec as Decoder>::Item, <Codec as Decoder>::Error>;
@ -96,7 +96,7 @@ where
impl<Io, St, Codec, Out> futures::Sink<<Codec as Encoder>::Item> impl<Io, St, Codec, Out> futures::Sink<<Codec as Encoder>::Item>
for ConnectResult<Io, St, Codec, Out> for ConnectResult<Io, St, Codec, Out>
where where
Io: AsyncRead + AsyncWrite, Io: AsyncRead + AsyncWrite + Unpin,
Codec: Encoder + Decoder, Codec: Encoder + Decoder,
{ {
type Error = <Codec as Encoder>::Error; type Error = <Codec as Encoder>::Error;

View file

@ -86,7 +86,7 @@ where
S: Service<Request = Request<U>, Response = Option<Response<U>>>, S: Service<Request = Request<U>, Response = Option<Response<U>>>,
S::Error: 'static, S::Error: 'static,
S::Future: 'static, S::Future: 'static,
T: AsyncRead + AsyncWrite, T: AsyncRead + AsyncWrite + Unpin,
U: Decoder + Encoder, U: Decoder + Encoder,
<U as Encoder>::Item: 'static, <U as Encoder>::Item: 'static,
<U as Encoder>::Error: std::fmt::Debug, <U as Encoder>::Error: std::fmt::Debug,

View file

@ -33,7 +33,7 @@ where
Response = ConnectResult<Io, St, Codec, Out>, Response = ConnectResult<Io, St, Codec, Out>,
>, >,
C::Error: fmt::Debug, C::Error: fmt::Debug,
Io: AsyncRead + AsyncWrite, Io: AsyncRead + AsyncWrite + Unpin,
Codec: Decoder + Encoder, Codec: Decoder + Encoder,
<Codec as Encoder>::Item: 'static, <Codec as Encoder>::Item: 'static,
<Codec as Encoder>::Error: std::fmt::Debug, <Codec as Encoder>::Error: std::fmt::Debug,
@ -79,7 +79,7 @@ pub struct FactoryBuilder<St, C, Io, Codec, Out> {
impl<St, C, Io, Codec, Out> FactoryBuilder<St, C, Io, Codec, Out> impl<St, C, Io, Codec, Out> FactoryBuilder<St, C, Io, Codec, Out>
where where
Io: AsyncRead + AsyncWrite, Io: AsyncRead + AsyncWrite + Unpin,
C: ServiceFactory< C: ServiceFactory<
Config = (), Config = (),
Request = Connect<Io, Codec>, Request = Connect<Io, Codec>,
@ -132,7 +132,7 @@ pub struct FramedService<St, C, T, Io, Codec, Out, Cfg> {
impl<St, C, T, Io, Codec, Out, Cfg> ServiceFactory impl<St, C, T, Io, Codec, Out, Cfg> ServiceFactory
for FramedService<St, C, T, Io, Codec, Out, Cfg> for FramedService<St, C, T, Io, Codec, Out, Cfg>
where where
Io: AsyncRead + AsyncWrite, Io: AsyncRead + AsyncWrite + Unpin,
C: ServiceFactory< C: ServiceFactory<
Config = (), Config = (),
Request = Connect<Io, Codec>, Request = Connect<Io, Codec>,
@ -173,7 +173,7 @@ where
#[pin_project::pin_project] #[pin_project::pin_project]
pub struct FramedServiceResponse<St, C, T, Io, Codec, Out> pub struct FramedServiceResponse<St, C, T, Io, Codec, Out>
where where
Io: AsyncRead + AsyncWrite, Io: AsyncRead + AsyncWrite + Unpin,
C: ServiceFactory< C: ServiceFactory<
Config = (), Config = (),
Request = Connect<Io, Codec>, Request = Connect<Io, Codec>,
@ -201,7 +201,7 @@ where
impl<St, C, T, Io, Codec, Out> Future for FramedServiceResponse<St, C, T, Io, Codec, Out> impl<St, C, T, Io, Codec, Out> Future for FramedServiceResponse<St, C, T, Io, Codec, Out>
where where
Io: AsyncRead + AsyncWrite, Io: AsyncRead + AsyncWrite + Unpin,
C: ServiceFactory< C: ServiceFactory<
Config = (), Config = (),
Request = Connect<Io, Codec>, Request = Connect<Io, Codec>,
@ -245,7 +245,7 @@ pub struct FramedServiceImpl<St, C, T, Io, Codec, Out> {
impl<St, C, T, Io, Codec, Out> Service for FramedServiceImpl<St, C, T, Io, Codec, Out> impl<St, C, T, Io, Codec, Out> Service for FramedServiceImpl<St, C, T, Io, Codec, Out>
where where
Io: AsyncRead + AsyncWrite, Io: AsyncRead + AsyncWrite + Unpin,
C: Service< C: Service<
Request = Connect<Io, Codec>, Request = Connect<Io, Codec>,
Response = ConnectResult<Io, St, Codec, Out>, Response = ConnectResult<Io, St, Codec, Out>,
@ -309,7 +309,7 @@ where
>, >,
<T::Service as Service>::Error: 'static, <T::Service as Service>::Error: 'static,
<T::Service as Service>::Future: 'static, <T::Service as Service>::Future: 'static,
Io: AsyncRead + AsyncWrite, Io: AsyncRead + AsyncWrite + Unpin,
Codec: Encoder + Decoder, Codec: Encoder + Decoder,
<Codec as Encoder>::Item: 'static, <Codec as Encoder>::Item: 'static,
<Codec as Encoder>::Error: std::fmt::Debug, <Codec as Encoder>::Error: std::fmt::Debug,
@ -336,7 +336,7 @@ where
>, >,
<T::Service as Service>::Error: 'static, <T::Service as Service>::Error: 'static,
<T::Service as Service>::Future: 'static, <T::Service as Service>::Future: 'static,
Io: AsyncRead + AsyncWrite, Io: AsyncRead + AsyncWrite + Unpin,
Codec: Encoder + Decoder, Codec: Encoder + Decoder,
<Codec as Encoder>::Item: 'static, <Codec as Encoder>::Item: 'static,
<Codec as Encoder>::Error: std::fmt::Debug, <Codec as Encoder>::Error: std::fmt::Debug,
@ -376,7 +376,7 @@ where
>, >,
<T::Service as Service>::Error: 'static, <T::Service as Service>::Error: 'static,
<T::Service as Service>::Future: 'static, <T::Service as Service>::Future: 'static,
Io: AsyncRead + AsyncWrite, Io: AsyncRead + AsyncWrite + Unpin,
Codec: Encoder + Decoder, Codec: Encoder + Decoder,
<Codec as Encoder>::Item: 'static, <Codec as Encoder>::Item: 'static,
<Codec as Encoder>::Error: std::fmt::Debug, <Codec as Encoder>::Error: std::fmt::Debug,
@ -403,7 +403,7 @@ where
>, >,
<T::Service as Service>::Error: 'static, <T::Service as Service>::Error: 'static,
<T::Service as Service>::Future: 'static, <T::Service as Service>::Future: 'static,
Io: AsyncRead + AsyncWrite, Io: AsyncRead + AsyncWrite + Unpin,
Codec: Encoder + Decoder, Codec: Encoder + Decoder,
<Codec as Encoder>::Item: 'static, <Codec as Encoder>::Item: 'static,
<Codec as Encoder>::Error: std::fmt::Debug, <Codec as Encoder>::Error: std::fmt::Debug,

View file

@ -29,7 +29,7 @@ where
S: Service<Request = Request<U>, Response = Response<U>>, S: Service<Request = Request<U>, Response = Response<U>>,
S::Error: 'static, S::Error: 'static,
S::Future: 'static, S::Future: 'static,
T: AsyncRead + AsyncWrite, T: AsyncRead + AsyncWrite + Unpin,
U: Encoder + Decoder, U: Encoder + Decoder,
<U as Encoder>::Item: 'static, <U as Encoder>::Item: 'static,
<U as Encoder>::Error: std::fmt::Debug, <U as Encoder>::Error: std::fmt::Debug,
@ -70,7 +70,7 @@ where
S: Service<Request = Request<U>, Response = Response<U>>, S: Service<Request = Request<U>, Response = Response<U>>,
S::Error: 'static, S::Error: 'static,
S::Future: 'static, S::Future: 'static,
T: AsyncRead + AsyncWrite, T: AsyncRead + AsyncWrite + Unpin,
U: Decoder + Encoder, U: Decoder + Encoder,
<U as Encoder>::Item: 'static, <U as Encoder>::Item: 'static,
<U as Encoder>::Error: std::fmt::Debug, <U as Encoder>::Error: std::fmt::Debug,
@ -130,16 +130,7 @@ where
&mut self.framed &mut self.framed
} }
fn poll_read(&mut self, cx: &mut Context<'_>) -> bool fn poll_read(&mut self, cx: &mut Context<'_>) -> bool {
where
S: Service<Request = Request<U>, Response = Response<U>>,
S::Error: 'static,
S::Future: 'static,
T: AsyncRead + AsyncWrite,
U: Decoder + Encoder,
<U as Encoder>::Item: 'static,
<U as Encoder>::Error: std::fmt::Debug,
{
loop { loop {
match self.service.poll_ready(cx) { match self.service.poll_ready(cx) {
Poll::Ready(Ok(_)) => { Poll::Ready(Ok(_)) => {
@ -171,16 +162,7 @@ where
} }
/// write to framed object /// write to framed object
fn poll_write(&mut self, cx: &mut Context<'_>) -> bool fn poll_write(&mut self, cx: &mut Context<'_>) -> bool {
where
S: Service<Request = Request<U>, Response = Response<U>>,
S::Error: 'static,
S::Future: 'static,
T: AsyncRead + AsyncWrite,
U: Decoder + Encoder,
<U as Encoder>::Item: 'static,
<U as Encoder>::Error: std::fmt::Debug,
{
loop { loop {
while !self.framed.is_write_buf_full() { while !self.framed.is_write_buf_full() {
match Pin::new(&mut self.rx).poll_next(cx) { match Pin::new(&mut self.rx).poll_next(cx) {
@ -226,7 +208,7 @@ where
S: Service<Request = Request<U>, Response = Response<U>>, S: Service<Request = Request<U>, Response = Response<U>>,
S::Error: 'static, S::Error: 'static,
S::Future: 'static, S::Future: 'static,
T: AsyncRead + AsyncWrite, T: AsyncRead + AsyncWrite + Unpin,
U: Decoder + Encoder, U: Decoder + Encoder,
<U as Encoder>::Item: 'static, <U as Encoder>::Item: 'static,
<U as Encoder>::Error: std::fmt::Debug, <U as Encoder>::Error: std::fmt::Debug,

View file

@ -41,7 +41,9 @@ pub trait Connection {
fn open_tunnel<H: Into<RequestHeadType>>(self, head: H) -> Self::TunnelFuture; fn open_tunnel<H: Into<RequestHeadType>>(self, head: H) -> Self::TunnelFuture;
} }
pub(super) trait ConnectionLifetime: AsyncRead + AsyncWrite + 'static { pub(super) trait ConnectionLifetime:
AsyncRead + AsyncWrite + Unpin + 'static
{
/// Close connection /// Close connection
fn close(&mut self); fn close(&mut self);

View file

@ -33,7 +33,7 @@ where
impl<T, B> Future for SendResponse<T, B> impl<T, B> Future for SendResponse<T, B>
where where
T: AsyncRead + AsyncWrite, T: AsyncRead + AsyncWrite + Unpin,
B: MessageBody, B: MessageBody,
{ {
type Output = Result<Framed<T, Codec>, Box<dyn Error>>; type Output = Result<Framed<T, Codec>, Box<dyn Error>>;

View file

@ -76,7 +76,7 @@ where
S: Service<Request = Request<U>, Response = Response<U>>, S: Service<Request = Request<U>, Response = Response<U>>,
S::Error: 'static, S::Error: 'static,
S::Future: 'static, S::Future: 'static,
T: AsyncRead + AsyncWrite, T: AsyncRead + AsyncWrite + Unpin,
U: Encoder + Decoder, U: Encoder + Decoder,
<U as Encoder>::Item: 'static, <U as Encoder>::Item: 'static,
<U as Encoder>::Error: std::fmt::Debug, <U as Encoder>::Error: std::fmt::Debug,
@ -117,7 +117,7 @@ where
S: Service<Request = Request<U>, Response = Response<U>>, S: Service<Request = Request<U>, Response = Response<U>>,
S::Error: 'static, S::Error: 'static,
S::Future: 'static, S::Future: 'static,
T: AsyncRead + AsyncWrite, T: AsyncRead + AsyncWrite + Unpin,
U: Decoder + Encoder, U: Decoder + Encoder,
<U as Encoder>::Item: 'static, <U as Encoder>::Item: 'static,
<U as Encoder>::Error: std::fmt::Debug, <U as Encoder>::Error: std::fmt::Debug,
@ -177,16 +177,7 @@ where
&mut self.framed &mut self.framed
} }
fn poll_read(&mut self, cx: &mut Context<'_>) -> bool fn poll_read(&mut self, cx: &mut Context<'_>) -> bool {
where
S: Service<Request = Request<U>, Response = Response<U>>,
S::Error: 'static,
S::Future: 'static,
T: AsyncRead + AsyncWrite,
U: Decoder + Encoder,
<U as Encoder>::Item: 'static,
<U as Encoder>::Error: std::fmt::Debug,
{
loop { loop {
match self.service.poll_ready(cx) { match self.service.poll_ready(cx) {
Poll::Ready(Ok(_)) => { Poll::Ready(Ok(_)) => {
@ -219,16 +210,7 @@ where
} }
/// write to framed object /// write to framed object
fn poll_write(&mut self, cx: &mut Context<'_>) -> bool fn poll_write(&mut self, cx: &mut Context<'_>) -> bool {
where
S: Service<Request = Request<U>, Response = Response<U>>,
S::Error: 'static,
S::Future: 'static,
T: AsyncRead + AsyncWrite,
U: Decoder + Encoder,
<U as Encoder>::Item: 'static,
<U as Encoder>::Error: std::fmt::Debug,
{
loop { loop {
while !self.framed.is_write_buf_full() { while !self.framed.is_write_buf_full() {
match Pin::new(&mut self.rx).poll_next(cx) { match Pin::new(&mut self.rx).poll_next(cx) {
@ -275,7 +257,7 @@ where
S: Service<Request = Request<U>, Response = Response<U>>, S: Service<Request = Request<U>, Response = Response<U>>,
S::Error: 'static, S::Error: 'static,
S::Future: 'static, S::Future: 'static,
T: AsyncRead + AsyncWrite, T: AsyncRead + AsyncWrite + Unpin,
U: Decoder + Encoder, U: Decoder + Encoder,
<U as Encoder>::Item: 'static, <U as Encoder>::Item: 'static,
<U as Encoder>::Error: std::fmt::Debug, <U as Encoder>::Error: std::fmt::Debug,

View file

@ -12,14 +12,14 @@ use super::{Codec, Frame, Message};
pub struct Dispatcher<S, T> pub struct Dispatcher<S, T>
where where
S: Service<Request = Frame, Response = Message> + 'static, S: Service<Request = Frame, Response = Message> + 'static,
T: AsyncRead + AsyncWrite, T: AsyncRead + AsyncWrite + Unpin,
{ {
inner: framed::Dispatcher<S, T, Codec>, inner: framed::Dispatcher<S, T, Codec>,
} }
impl<S, T> Dispatcher<S, T> impl<S, T> Dispatcher<S, T>
where where
T: AsyncRead + AsyncWrite, T: AsyncRead + AsyncWrite + Unpin,
S: Service<Request = Frame, Response = Message>, S: Service<Request = Frame, Response = Message>,
S::Future: 'static, S::Future: 'static,
S::Error: 'static, S::Error: 'static,
@ -39,7 +39,7 @@ where
impl<S, T> Future for Dispatcher<S, T> impl<S, T> Future for Dispatcher<S, T>
where where
T: AsyncRead + AsyncWrite, T: AsyncRead + AsyncWrite + Unpin,
S: Service<Request = Frame, Response = Message>, S: Service<Request = Frame, Response = Message>,
S::Future: 'static, S::Future: 'static,
S::Error: 'static, S::Error: 'static,