mirror of https://github.com/fafhrd91/actix-web
feat(http,web): expose config for H2 window sizes (#3926)
* feat(http,web): expose config for H2 window sizes * changelog
This commit is contained in:
parent
6d81907540
commit
5a6c8d235b
|
|
@ -3,10 +3,13 @@
|
||||||
## Unreleased
|
## Unreleased
|
||||||
|
|
||||||
- Minimum supported Rust version (MSRV) is now 1.88.
|
- Minimum supported Rust version (MSRV) is now 1.88.
|
||||||
|
- Increase default HTTP/2 flow control window sizes. [#3638]
|
||||||
|
- Expose configuration methods to improve upload throughput. [#3638]
|
||||||
- Fix truncated body ending without error when connection closed abnormally. [#3067]
|
- Fix truncated body ending without error when connection closed abnormally. [#3067]
|
||||||
- Add config/method for `TCP_NODELAY`. [#3918]
|
- Add config/method for `TCP_NODELAY`. [#3918]
|
||||||
- Do not compress 206 Partial Content responses. [#3191]
|
- Do not compress 206 Partial Content responses. [#3191]
|
||||||
|
|
||||||
|
[#3638]: https://github.com/actix/actix-web/issues/3638
|
||||||
[#3067]: https://github.com/actix/actix-web/pull/3067
|
[#3067]: https://github.com/actix/actix-web/pull/3067
|
||||||
[#3918]: https://github.com/actix/actix-web/pull/3918
|
[#3918]: https://github.com/actix/actix-web/pull/3918
|
||||||
[#3191]: https://github.com/actix/actix-web/issues/3191
|
[#3191]: https://github.com/actix/actix-web/issues/3191
|
||||||
|
|
|
||||||
|
|
@ -5,6 +5,7 @@ use actix_service::{IntoServiceFactory, Service, ServiceFactory};
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
body::{BoxBody, MessageBody},
|
body::{BoxBody, MessageBody},
|
||||||
|
config::{DEFAULT_H2_CONN_WINDOW_SIZE, DEFAULT_H2_STREAM_WINDOW_SIZE},
|
||||||
h1::{self, ExpectHandler, H1Service, UpgradeHandler},
|
h1::{self, ExpectHandler, H1Service, UpgradeHandler},
|
||||||
service::HttpService,
|
service::HttpService,
|
||||||
ConnectCallback, Extensions, KeepAlive, Request, Response, ServiceConfigBuilder,
|
ConnectCallback, Extensions, KeepAlive, Request, Response, ServiceConfigBuilder,
|
||||||
|
|
@ -21,6 +22,8 @@ pub struct HttpServiceBuilder<T, S, X = ExpectHandler, U = UpgradeHandler> {
|
||||||
secure: bool,
|
secure: bool,
|
||||||
local_addr: Option<net::SocketAddr>,
|
local_addr: Option<net::SocketAddr>,
|
||||||
h1_allow_half_closed: bool,
|
h1_allow_half_closed: bool,
|
||||||
|
h2_conn_window_size: u32,
|
||||||
|
h2_stream_window_size: u32,
|
||||||
expect: X,
|
expect: X,
|
||||||
upgrade: Option<U>,
|
upgrade: Option<U>,
|
||||||
on_connect_ext: Option<Rc<ConnectCallback<T>>>,
|
on_connect_ext: Option<Rc<ConnectCallback<T>>>,
|
||||||
|
|
@ -44,6 +47,8 @@ where
|
||||||
secure: false,
|
secure: false,
|
||||||
local_addr: None,
|
local_addr: None,
|
||||||
h1_allow_half_closed: true,
|
h1_allow_half_closed: true,
|
||||||
|
h2_conn_window_size: DEFAULT_H2_CONN_WINDOW_SIZE,
|
||||||
|
h2_stream_window_size: DEFAULT_H2_STREAM_WINDOW_SIZE,
|
||||||
|
|
||||||
// dispatcher parts
|
// dispatcher parts
|
||||||
expect: ExpectHandler,
|
expect: ExpectHandler,
|
||||||
|
|
@ -146,6 +151,22 @@ where
|
||||||
self
|
self
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Sets initial stream-level flow control window size for HTTP/2 connections.
|
||||||
|
///
|
||||||
|
/// See [`ServiceConfigBuilder::h2_initial_window_size`] for more details.
|
||||||
|
pub fn h2_initial_window_size(mut self, size: u32) -> Self {
|
||||||
|
self.h2_stream_window_size = size;
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Sets initial connection-level flow control window size for HTTP/2 connections.
|
||||||
|
///
|
||||||
|
/// See [`ServiceConfigBuilder::h2_initial_connection_window_size`] for more details.
|
||||||
|
pub fn h2_initial_connection_window_size(mut self, size: u32) -> Self {
|
||||||
|
self.h2_conn_window_size = size;
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
/// Provide service for `EXPECT: 100-Continue` support.
|
/// Provide service for `EXPECT: 100-Continue` support.
|
||||||
///
|
///
|
||||||
/// Service get called with request that contains `EXPECT` header.
|
/// Service get called with request that contains `EXPECT` header.
|
||||||
|
|
@ -166,6 +187,8 @@ where
|
||||||
secure: self.secure,
|
secure: self.secure,
|
||||||
local_addr: self.local_addr,
|
local_addr: self.local_addr,
|
||||||
h1_allow_half_closed: self.h1_allow_half_closed,
|
h1_allow_half_closed: self.h1_allow_half_closed,
|
||||||
|
h2_conn_window_size: self.h2_conn_window_size,
|
||||||
|
h2_stream_window_size: self.h2_stream_window_size,
|
||||||
expect: expect.into_factory(),
|
expect: expect.into_factory(),
|
||||||
upgrade: self.upgrade,
|
upgrade: self.upgrade,
|
||||||
on_connect_ext: self.on_connect_ext,
|
on_connect_ext: self.on_connect_ext,
|
||||||
|
|
@ -192,6 +215,8 @@ where
|
||||||
secure: self.secure,
|
secure: self.secure,
|
||||||
local_addr: self.local_addr,
|
local_addr: self.local_addr,
|
||||||
h1_allow_half_closed: self.h1_allow_half_closed,
|
h1_allow_half_closed: self.h1_allow_half_closed,
|
||||||
|
h2_conn_window_size: self.h2_conn_window_size,
|
||||||
|
h2_stream_window_size: self.h2_stream_window_size,
|
||||||
expect: self.expect,
|
expect: self.expect,
|
||||||
upgrade: Some(upgrade.into_factory()),
|
upgrade: Some(upgrade.into_factory()),
|
||||||
on_connect_ext: self.on_connect_ext,
|
on_connect_ext: self.on_connect_ext,
|
||||||
|
|
@ -229,6 +254,8 @@ where
|
||||||
.secure(self.secure)
|
.secure(self.secure)
|
||||||
.local_addr(self.local_addr)
|
.local_addr(self.local_addr)
|
||||||
.h1_allow_half_closed(self.h1_allow_half_closed)
|
.h1_allow_half_closed(self.h1_allow_half_closed)
|
||||||
|
.h2_initial_window_size(self.h2_stream_window_size)
|
||||||
|
.h2_initial_connection_window_size(self.h2_conn_window_size)
|
||||||
.build();
|
.build();
|
||||||
|
|
||||||
H1Service::with_config(cfg, service.into_factory())
|
H1Service::with_config(cfg, service.into_factory())
|
||||||
|
|
@ -256,6 +283,8 @@ where
|
||||||
.secure(self.secure)
|
.secure(self.secure)
|
||||||
.local_addr(self.local_addr)
|
.local_addr(self.local_addr)
|
||||||
.h1_allow_half_closed(self.h1_allow_half_closed)
|
.h1_allow_half_closed(self.h1_allow_half_closed)
|
||||||
|
.h2_initial_window_size(self.h2_stream_window_size)
|
||||||
|
.h2_initial_connection_window_size(self.h2_conn_window_size)
|
||||||
.build();
|
.build();
|
||||||
|
|
||||||
crate::h2::H2Service::with_config(cfg, service.into_factory())
|
crate::h2::H2Service::with_config(cfg, service.into_factory())
|
||||||
|
|
@ -280,6 +309,8 @@ where
|
||||||
.secure(self.secure)
|
.secure(self.secure)
|
||||||
.local_addr(self.local_addr)
|
.local_addr(self.local_addr)
|
||||||
.h1_allow_half_closed(self.h1_allow_half_closed)
|
.h1_allow_half_closed(self.h1_allow_half_closed)
|
||||||
|
.h2_initial_window_size(self.h2_stream_window_size)
|
||||||
|
.h2_initial_connection_window_size(self.h2_conn_window_size)
|
||||||
.build();
|
.build();
|
||||||
|
|
||||||
HttpService::with_config(cfg, service.into_factory())
|
HttpService::with_config(cfg, service.into_factory())
|
||||||
|
|
|
||||||
|
|
@ -8,6 +8,16 @@ use bytes::BytesMut;
|
||||||
|
|
||||||
use crate::{date::DateService, KeepAlive};
|
use crate::{date::DateService, KeepAlive};
|
||||||
|
|
||||||
|
/// Default HTTP/2 initial connection-level flow control window size.
|
||||||
|
///
|
||||||
|
/// Matches awc's defaults to avoid poor throughput on high-BDP links.
|
||||||
|
pub(crate) const DEFAULT_H2_CONN_WINDOW_SIZE: u32 = 1024 * 1024 * 2; // 2MiB
|
||||||
|
|
||||||
|
/// Default HTTP/2 initial stream-level flow control window size.
|
||||||
|
///
|
||||||
|
/// Matches awc's defaults to avoid poor throughput on high-BDP links.
|
||||||
|
pub(crate) const DEFAULT_H2_STREAM_WINDOW_SIZE: u32 = 1024 * 1024; // 1MiB
|
||||||
|
|
||||||
/// A builder for creating a [`ServiceConfig`]
|
/// A builder for creating a [`ServiceConfig`]
|
||||||
#[derive(Default, Debug)]
|
#[derive(Default, Debug)]
|
||||||
pub struct ServiceConfigBuilder {
|
pub struct ServiceConfigBuilder {
|
||||||
|
|
@ -76,6 +86,28 @@ impl ServiceConfigBuilder {
|
||||||
self
|
self
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Sets initial stream-level flow control window size for HTTP/2 connections.
|
||||||
|
///
|
||||||
|
/// Higher values can improve upload performance on high-latency links at the cost of higher
|
||||||
|
/// worst-case memory usage per connection.
|
||||||
|
///
|
||||||
|
/// The default value is 1MiB.
|
||||||
|
pub fn h2_initial_window_size(mut self, size: u32) -> Self {
|
||||||
|
self.inner.h2_stream_window_size = size;
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Sets initial connection-level flow control window size for HTTP/2 connections.
|
||||||
|
///
|
||||||
|
/// Higher values can improve upload performance on high-latency links at the cost of higher
|
||||||
|
/// worst-case memory usage per connection.
|
||||||
|
///
|
||||||
|
/// The default value is 2MiB.
|
||||||
|
pub fn h2_initial_connection_window_size(mut self, size: u32) -> Self {
|
||||||
|
self.inner.h2_conn_window_size = size;
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
/// Builds a [`ServiceConfig`] from this [`ServiceConfigBuilder`] instance
|
/// Builds a [`ServiceConfig`] from this [`ServiceConfigBuilder`] instance
|
||||||
pub fn build(self) -> ServiceConfig {
|
pub fn build(self) -> ServiceConfig {
|
||||||
ServiceConfig(Rc::new(self.inner))
|
ServiceConfig(Rc::new(self.inner))
|
||||||
|
|
@ -96,6 +128,8 @@ struct Inner {
|
||||||
tcp_nodelay: Option<bool>,
|
tcp_nodelay: Option<bool>,
|
||||||
date_service: DateService,
|
date_service: DateService,
|
||||||
h1_allow_half_closed: bool,
|
h1_allow_half_closed: bool,
|
||||||
|
h2_conn_window_size: u32,
|
||||||
|
h2_stream_window_size: u32,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Default for Inner {
|
impl Default for Inner {
|
||||||
|
|
@ -109,6 +143,8 @@ impl Default for Inner {
|
||||||
tcp_nodelay: None,
|
tcp_nodelay: None,
|
||||||
date_service: DateService::new(),
|
date_service: DateService::new(),
|
||||||
h1_allow_half_closed: true,
|
h1_allow_half_closed: true,
|
||||||
|
h2_conn_window_size: DEFAULT_H2_CONN_WINDOW_SIZE,
|
||||||
|
h2_stream_window_size: DEFAULT_H2_STREAM_WINDOW_SIZE,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
@ -131,6 +167,8 @@ impl ServiceConfig {
|
||||||
tcp_nodelay: None,
|
tcp_nodelay: None,
|
||||||
date_service: DateService::new(),
|
date_service: DateService::new(),
|
||||||
h1_allow_half_closed: true,
|
h1_allow_half_closed: true,
|
||||||
|
h2_conn_window_size: DEFAULT_H2_CONN_WINDOW_SIZE,
|
||||||
|
h2_stream_window_size: DEFAULT_H2_STREAM_WINDOW_SIZE,
|
||||||
}))
|
}))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -195,6 +233,16 @@ impl ServiceConfig {
|
||||||
self.0.tcp_nodelay
|
self.0.tcp_nodelay
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// HTTP/2 initial stream-level flow control window size (in bytes).
|
||||||
|
pub fn h2_initial_window_size(&self) -> u32 {
|
||||||
|
self.0.h2_stream_window_size
|
||||||
|
}
|
||||||
|
|
||||||
|
/// HTTP/2 initial connection-level flow control window size (in bytes).
|
||||||
|
pub fn h2_initial_connection_window_size(&self) -> u32 {
|
||||||
|
self.0.h2_conn_window_size
|
||||||
|
}
|
||||||
|
|
||||||
pub(crate) fn now(&self) -> Instant {
|
pub(crate) fn now(&self) -> Instant {
|
||||||
self.0.date_service.now()
|
self.0.date_service.now()
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -11,7 +11,7 @@ use actix_rt::time::{sleep_until, Sleep};
|
||||||
use bytes::Bytes;
|
use bytes::Bytes;
|
||||||
use futures_core::{ready, Stream};
|
use futures_core::{ready, Stream};
|
||||||
use h2::{
|
use h2::{
|
||||||
server::{handshake, Connection, Handshake},
|
server::{Builder, Connection, Handshake},
|
||||||
RecvStream,
|
RecvStream,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
@ -61,8 +61,13 @@ pub(crate) fn handshake_with_timeout<T>(io: T, config: &ServiceConfig) -> Handsh
|
||||||
where
|
where
|
||||||
T: AsyncRead + AsyncWrite + Unpin,
|
T: AsyncRead + AsyncWrite + Unpin,
|
||||||
{
|
{
|
||||||
|
let mut builder = Builder::new();
|
||||||
|
builder
|
||||||
|
.initial_window_size(config.h2_initial_window_size())
|
||||||
|
.initial_connection_window_size(config.h2_initial_connection_window_size());
|
||||||
|
|
||||||
HandshakeWithTimeout {
|
HandshakeWithTimeout {
|
||||||
handshake: handshake(io),
|
handshake: builder.handshake(io),
|
||||||
timer: config
|
timer: config
|
||||||
.client_request_deadline()
|
.client_request_deadline()
|
||||||
.map(|deadline| Box::pin(sleep_until(deadline.into()))),
|
.map(|deadline| Box::pin(sleep_until(deadline.into()))),
|
||||||
|
|
|
||||||
|
|
@ -10,7 +10,10 @@ use actix_http::{
|
||||||
header, Error, HttpService, KeepAlive, Request, Response, StatusCode, Version,
|
header, Error, HttpService, KeepAlive, Request, Response, StatusCode, Version,
|
||||||
};
|
};
|
||||||
use actix_http_test::test_server;
|
use actix_http_test::test_server;
|
||||||
use actix_rt::{net::TcpStream, time::sleep};
|
use actix_rt::{
|
||||||
|
net::TcpStream,
|
||||||
|
time::{sleep, timeout},
|
||||||
|
};
|
||||||
use actix_service::fn_service;
|
use actix_service::fn_service;
|
||||||
use actix_utils::future::{err, ok, ready};
|
use actix_utils::future::{err, ok, ready};
|
||||||
use bytes::Bytes;
|
use bytes::Bytes;
|
||||||
|
|
@ -953,3 +956,62 @@ async fn h2c_auto() {
|
||||||
|
|
||||||
srv.stop().await;
|
srv.stop().await;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[actix_rt::test]
|
||||||
|
async fn h2_flow_control_window_sizes() {
|
||||||
|
let mut srv = test_server(|| {
|
||||||
|
HttpService::build()
|
||||||
|
.keep_alive(KeepAlive::Disabled)
|
||||||
|
.finish(|_req: Request| ok::<_, Infallible>(Response::ok()))
|
||||||
|
.tcp_auto_h2c()
|
||||||
|
})
|
||||||
|
.await;
|
||||||
|
|
||||||
|
let tcp = TcpStream::connect(srv.addr()).await.unwrap();
|
||||||
|
|
||||||
|
let mut builder = h2::client::Builder::new();
|
||||||
|
builder.max_send_buffer_size(4 * 1024 * 1024);
|
||||||
|
|
||||||
|
let (h2, connection) = builder.handshake(tcp).await.unwrap();
|
||||||
|
tokio::spawn(async move { connection.await.unwrap() });
|
||||||
|
let mut h2 = h2.ready().await.unwrap();
|
||||||
|
|
||||||
|
let request = ::http::Request::builder()
|
||||||
|
.method("POST")
|
||||||
|
.uri("/")
|
||||||
|
.body(())
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
let (response, mut send) = h2.send_request(request, false).unwrap();
|
||||||
|
|
||||||
|
// request more than the default 64KiB. if server is advertising larger flow control windows,
|
||||||
|
// we should get at least 1MiB assigned.
|
||||||
|
send.reserve_capacity(2 * 1024 * 1024);
|
||||||
|
|
||||||
|
let cap = timeout(Duration::from_secs(2), async {
|
||||||
|
loop {
|
||||||
|
let cap = std::future::poll_fn(|cx| send.poll_capacity(cx))
|
||||||
|
.await
|
||||||
|
.unwrap()
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
if cap >= 1024 * 1024 {
|
||||||
|
break cap;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.await
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
assert!(
|
||||||
|
cap >= 1024 * 1024,
|
||||||
|
"expected >= 1MiB send capacity, got {cap}"
|
||||||
|
);
|
||||||
|
|
||||||
|
send.send_data(Bytes::new(), true).unwrap();
|
||||||
|
|
||||||
|
let res = response.await.unwrap();
|
||||||
|
assert!(res.status().is_success());
|
||||||
|
|
||||||
|
srv.stop().await;
|
||||||
|
}
|
||||||
|
|
|
||||||
|
|
@ -3,6 +3,8 @@
|
||||||
## Unreleased
|
## Unreleased
|
||||||
|
|
||||||
- Minimum supported Rust version (MSRV) is now 1.88.
|
- Minimum supported Rust version (MSRV) is now 1.88.
|
||||||
|
- Improve HTTP/2 upload throughput by increasing default flow control window sizes. [#3638]
|
||||||
|
- Add `HttpServer::{h2_initial_window_size, h2_initial_connection_window_size}` methods for tuning. [#3638]
|
||||||
- Add `HttpRequest::url_for_map` and `HttpRequest::url_for_iter` methods for named URL parameters. [#3895]
|
- Add `HttpRequest::url_for_map` and `HttpRequest::url_for_iter` methods for named URL parameters. [#3895]
|
||||||
- Ignore unparsable cookies in `Cookie` request header.
|
- Ignore unparsable cookies in `Cookie` request header.
|
||||||
- Add `experimental-introspection` feature to report configured routes [#3594]
|
- Add `experimental-introspection` feature to report configured routes [#3594]
|
||||||
|
|
@ -13,6 +15,7 @@
|
||||||
[#3895]: https://github.com/actix/actix-web/pull/3895
|
[#3895]: https://github.com/actix/actix-web/pull/3895
|
||||||
[#3594]: https://github.com/actix/actix-web/pull/3594
|
[#3594]: https://github.com/actix/actix-web/pull/3594
|
||||||
[#3918]: https://github.com/actix/actix-web/pull/3918
|
[#3918]: https://github.com/actix/actix-web/pull/3918
|
||||||
|
[#3638]: https://github.com/actix/actix-web/issues/3638
|
||||||
[#3562]: https://github.com/actix/actix-web/issues/3562
|
[#3562]: https://github.com/actix/actix-web/issues/3562
|
||||||
[#3191]: https://github.com/actix/actix-web/issues/3191
|
[#3191]: https://github.com/actix/actix-web/issues/3191
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -33,6 +33,8 @@ struct Config {
|
||||||
client_request_timeout: Duration,
|
client_request_timeout: Duration,
|
||||||
client_disconnect_timeout: Duration,
|
client_disconnect_timeout: Duration,
|
||||||
h1_allow_half_closed: bool,
|
h1_allow_half_closed: bool,
|
||||||
|
h2_initial_window_size: Option<u32>,
|
||||||
|
h2_initial_connection_window_size: Option<u32>,
|
||||||
#[allow(dead_code)] // only dead when no TLS features are enabled
|
#[allow(dead_code)] // only dead when no TLS features are enabled
|
||||||
tls_handshake_timeout: Option<Duration>,
|
tls_handshake_timeout: Option<Duration>,
|
||||||
}
|
}
|
||||||
|
|
@ -120,6 +122,8 @@ where
|
||||||
client_request_timeout: Duration::from_secs(5),
|
client_request_timeout: Duration::from_secs(5),
|
||||||
client_disconnect_timeout: Duration::from_secs(1),
|
client_disconnect_timeout: Duration::from_secs(1),
|
||||||
h1_allow_half_closed: true,
|
h1_allow_half_closed: true,
|
||||||
|
h2_initial_window_size: None,
|
||||||
|
h2_initial_connection_window_size: None,
|
||||||
tls_handshake_timeout: None,
|
tls_handshake_timeout: None,
|
||||||
})),
|
})),
|
||||||
backlog: 1024,
|
backlog: 1024,
|
||||||
|
|
@ -282,6 +286,33 @@ where
|
||||||
self
|
self
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Sets initial stream-level flow control window size for HTTP/2 connections.
|
||||||
|
///
|
||||||
|
/// Higher values can improve upload performance on high-latency links at the cost of higher
|
||||||
|
/// worst-case memory usage per connection.
|
||||||
|
///
|
||||||
|
/// The default value is 1MiB.
|
||||||
|
#[cfg(feature = "http2")]
|
||||||
|
pub fn h2_initial_window_size(self, size: u32) -> Self {
|
||||||
|
self.config.lock().unwrap().h2_initial_window_size = Some(size);
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Sets initial connection-level flow control window size for HTTP/2 connections.
|
||||||
|
///
|
||||||
|
/// Higher values can improve upload performance on high-latency links at the cost of higher
|
||||||
|
/// worst-case memory usage per connection.
|
||||||
|
///
|
||||||
|
/// The default value is 2MiB.
|
||||||
|
#[cfg(feature = "http2")]
|
||||||
|
pub fn h2_initial_connection_window_size(self, size: u32) -> Self {
|
||||||
|
self.config
|
||||||
|
.lock()
|
||||||
|
.unwrap()
|
||||||
|
.h2_initial_connection_window_size = Some(size);
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
/// Sets function that will be called once before each connection is handled.
|
/// Sets function that will be called once before each connection is handled.
|
||||||
///
|
///
|
||||||
/// It will receive a `&std::any::Any`, which contains underlying connection type and an
|
/// It will receive a `&std::any::Any`, which contains underlying connection type and an
|
||||||
|
|
@ -590,6 +621,14 @@ where
|
||||||
svc = svc.tcp_nodelay(enabled);
|
svc = svc.tcp_nodelay(enabled);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if let Some(val) = cfg.h2_initial_window_size {
|
||||||
|
svc = svc.h2_initial_window_size(val);
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Some(val) = cfg.h2_initial_connection_window_size {
|
||||||
|
svc = svc.h2_initial_connection_window_size(val);
|
||||||
|
}
|
||||||
|
|
||||||
if let Some(handler) = on_connect_fn.clone() {
|
if let Some(handler) = on_connect_fn.clone() {
|
||||||
svc =
|
svc =
|
||||||
svc.on_connect_ext(move |io: &_, ext: _| (handler)(io as &dyn Any, ext))
|
svc.on_connect_ext(move |io: &_, ext: _| (handler)(io as &dyn Any, ext))
|
||||||
|
|
@ -639,6 +678,14 @@ where
|
||||||
svc = svc.tcp_nodelay(enabled);
|
svc = svc.tcp_nodelay(enabled);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if let Some(val) = cfg.h2_initial_window_size {
|
||||||
|
svc = svc.h2_initial_window_size(val);
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Some(val) = cfg.h2_initial_connection_window_size {
|
||||||
|
svc = svc.h2_initial_connection_window_size(val);
|
||||||
|
}
|
||||||
|
|
||||||
if let Some(handler) = on_connect_fn.clone() {
|
if let Some(handler) = on_connect_fn.clone() {
|
||||||
svc =
|
svc =
|
||||||
svc.on_connect_ext(move |io: &_, ext: _| (handler)(io as &dyn Any, ext))
|
svc.on_connect_ext(move |io: &_, ext: _| (handler)(io as &dyn Any, ext))
|
||||||
|
|
@ -719,6 +766,14 @@ where
|
||||||
svc = svc.tcp_nodelay(enabled);
|
svc = svc.tcp_nodelay(enabled);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if let Some(val) = c.h2_initial_window_size {
|
||||||
|
svc = svc.h2_initial_window_size(val);
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Some(val) = c.h2_initial_connection_window_size {
|
||||||
|
svc = svc.h2_initial_connection_window_size(val);
|
||||||
|
}
|
||||||
|
|
||||||
if let Some(handler) = on_connect_fn.clone() {
|
if let Some(handler) = on_connect_fn.clone() {
|
||||||
svc = svc
|
svc = svc
|
||||||
.on_connect_ext(move |io: &_, ext: _| (handler)(io as &dyn Any, ext));
|
.on_connect_ext(move |io: &_, ext: _| (handler)(io as &dyn Any, ext));
|
||||||
|
|
@ -774,6 +829,14 @@ where
|
||||||
svc = svc.tcp_nodelay(enabled);
|
svc = svc.tcp_nodelay(enabled);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if let Some(val) = c.h2_initial_window_size {
|
||||||
|
svc = svc.h2_initial_window_size(val);
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Some(val) = c.h2_initial_connection_window_size {
|
||||||
|
svc = svc.h2_initial_connection_window_size(val);
|
||||||
|
}
|
||||||
|
|
||||||
if let Some(handler) = on_connect_fn.clone() {
|
if let Some(handler) = on_connect_fn.clone() {
|
||||||
svc = svc
|
svc = svc
|
||||||
.on_connect_ext(move |io: &_, ext: _| (handler)(io as &dyn Any, ext));
|
.on_connect_ext(move |io: &_, ext: _| (handler)(io as &dyn Any, ext));
|
||||||
|
|
@ -844,6 +907,14 @@ where
|
||||||
svc = svc.tcp_nodelay(enabled);
|
svc = svc.tcp_nodelay(enabled);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if let Some(val) = c.h2_initial_window_size {
|
||||||
|
svc = svc.h2_initial_window_size(val);
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Some(val) = c.h2_initial_connection_window_size {
|
||||||
|
svc = svc.h2_initial_connection_window_size(val);
|
||||||
|
}
|
||||||
|
|
||||||
if let Some(handler) = on_connect_fn.clone() {
|
if let Some(handler) = on_connect_fn.clone() {
|
||||||
svc = svc
|
svc = svc
|
||||||
.on_connect_ext(move |io: &_, ext: _| (handler)(io as &dyn Any, ext));
|
.on_connect_ext(move |io: &_, ext: _| (handler)(io as &dyn Any, ext));
|
||||||
|
|
@ -914,6 +985,14 @@ where
|
||||||
svc = svc.tcp_nodelay(enabled);
|
svc = svc.tcp_nodelay(enabled);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if let Some(val) = c.h2_initial_window_size {
|
||||||
|
svc = svc.h2_initial_window_size(val);
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Some(val) = c.h2_initial_connection_window_size {
|
||||||
|
svc = svc.h2_initial_connection_window_size(val);
|
||||||
|
}
|
||||||
|
|
||||||
if let Some(handler) = on_connect_fn.clone() {
|
if let Some(handler) = on_connect_fn.clone() {
|
||||||
svc = svc
|
svc = svc
|
||||||
.on_connect_ext(move |io: &_, ext: _| (handler)(io as &dyn Any, ext));
|
.on_connect_ext(move |io: &_, ext: _| (handler)(io as &dyn Any, ext));
|
||||||
|
|
@ -985,6 +1064,14 @@ where
|
||||||
svc = svc.tcp_nodelay(enabled);
|
svc = svc.tcp_nodelay(enabled);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if let Some(val) = c.h2_initial_window_size {
|
||||||
|
svc = svc.h2_initial_window_size(val);
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Some(val) = c.h2_initial_connection_window_size {
|
||||||
|
svc = svc.h2_initial_connection_window_size(val);
|
||||||
|
}
|
||||||
|
|
||||||
if let Some(handler) = on_connect_fn.clone() {
|
if let Some(handler) = on_connect_fn.clone() {
|
||||||
svc = svc
|
svc = svc
|
||||||
.on_connect_ext(move |io: &_, ext: _| (handler)(io as &dyn Any, ext));
|
.on_connect_ext(move |io: &_, ext: _| (handler)(io as &dyn Any, ext));
|
||||||
|
|
|
||||||
Loading…
Reference in New Issue