tokio_quiche/settings/quic.rs
1// Copyright (C) 2025, Cloudflare, Inc.
2// All rights reserved.
3//
4// Redistribution and use in source and binary forms, with or without
5// modification, are permitted provided that the following conditions are
6// met:
7//
8// * Redistributions of source code must retain the above copyright notice,
9// this list of conditions and the following disclaimer.
10//
11// * Redistributions in binary form must reproduce the above copyright
12// notice, this list of conditions and the following disclaimer in the
13// documentation and/or other materials provided with the distribution.
14//
15// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS
16// IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
17// THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
18// PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR
19// CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
20// EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
21// PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
22// PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
23// LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
24// NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
25// SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
26
27use foundations::settings::settings;
28use serde_with::serde_as;
29use serde_with::DurationMilliSeconds;
30use std::time::Duration;
31
32/// QUIC configuration parameters.
33#[serde_as]
34#[settings]
35pub struct QuicSettings {
36 /// Configures the list of supported application protocols. Defaults to
37 /// `[b"h3"]`.
38 #[serde(skip, default = "QuicSettings::default_alpn")]
39 pub alpn: Vec<Vec<u8>>,
40
41 /// Configures whether to enable DATAGRAM frame support. H3 connections
42 /// copy this setting from the underlying QUIC connection.
43 ///
44 /// Defaults to `true`.
45 #[serde(default = "QuicSettings::default_enable_dgram")]
46 pub enable_dgram: bool,
47
48 /// Max queue length for received DATAGRAM frames. Defaults to `2^16`.
49 #[serde(default = "QuicSettings::default_dgram_max_queue_len")]
50 pub dgram_recv_max_queue_len: usize,
51
52 /// Max queue length for sending DATAGRAM frames. Defaults to `2^16`.
53 #[serde(default = "QuicSettings::default_dgram_max_queue_len")]
54 pub dgram_send_max_queue_len: usize,
55
56 /// Sets the `initial_max_data` transport parameter. Defaults to 10 MB.
57 #[serde(default = "QuicSettings::default_initial_max_data")]
58 pub initial_max_data: u64,
59
60 /// Sets the `initial_max_stream_data_bidi_local` transport parameter.
61 /// Defaults to 1 MB.
62 #[serde(default = "QuicSettings::default_initial_max_stream_data")]
63 pub initial_max_stream_data_bidi_local: u64,
64
65 /// Sets the `initial_max_stream_data_bidi_remote` transport parameter.
66 /// Defaults to 1 MB.
67 #[serde(default = "QuicSettings::default_initial_max_stream_data")]
68 pub initial_max_stream_data_bidi_remote: u64,
69
70 /// Sets the `initial_max_stream_data_uni` transport parameter.
71 /// Defaults to 1 MB.
72 #[serde(default = "QuicSettings::default_initial_max_stream_data")]
73 pub initial_max_stream_data_uni: u64,
74
75 /// Sets the `initial_max_streams_bidi` transport parameter.
76 /// Defaults to `100`.
77 #[serde(default = "QuicSettings::default_initial_max_streams")]
78 pub initial_max_streams_bidi: u64,
79
80 /// Sets the `initial_max_streams_uni` transport parameter.
81 /// Defaults to `100`.
82 #[serde(default = "QuicSettings::default_initial_max_streams")]
83 pub initial_max_streams_uni: u64,
84
85 /// Configures the max idle timeout of the connection in milliseconds. The
86 /// real idle timeout is the minimum of this and the peer's
87 /// `max_idle_timeout`.
88 ///
89 /// Defaults to 56 seconds.
90 #[serde(
91 rename = "max_idle_timeout_ms",
92 default = "QuicSettings::default_max_idle_timeout"
93 )]
94 #[serde_as(as = "Option<DurationMilliSeconds>")]
95 pub max_idle_timeout: Option<Duration>,
96
97 /// Configures whether the local endpoint supports active connection
98 /// migration. Defaults to `true` (meaning disabled).
99 #[serde(default = "QuicSettings::default_disable_active_migration")]
100 pub disable_active_migration: bool,
101
102 /// Sets the maximum incoming UDP payload size. Defaults to 1350 bytes.
103 #[serde(default = "QuicSettings::default_max_recv_udp_payload_size")]
104 pub max_recv_udp_payload_size: usize,
105
106 /// Sets the maximum outgoing UDP payload size. Defaults to 1350 bytes.
107 #[serde(default = "QuicSettings::default_max_send_udp_payload_size")]
108 pub max_send_udp_payload_size: usize,
109
110 /// Whether to validate client IPs in QUIC initials.
111 ///
112 /// If set to `true`, any received QUIC initial will immediately spawn a
113 /// connection and start crypto operations for the handshake. Otherwise,
114 /// the client is asked to execute a stateless retry first (the default).
115 pub disable_client_ip_validation: bool,
116
117 /// Path to a file in which TLS secrets will be logged in
118 /// [SSLKEYLOGFILE format](https://tlswg.org/sslkeylogfile/draft-ietf-tls-keylogfile.html).
119 pub keylog_file: Option<String>,
120
121 /// Path to a directory where QLOG files will be saved.
122 pub qlog_dir: Option<String>,
123
124 /// Congestion control algorithm to use.
125 ///
126 /// For available values, see
127 /// [`CongestionControlAlgorithm`](quiche::CongestionControlAlgorithm).
128 /// Defaults to `cubic`.
129 #[serde(default = "QuicSettings::default_cc_algorithm")]
130 pub cc_algorithm: String,
131
132 /// Whether to use HyStart++ (only with `cubic` and `reno` CC). Defaults to
133 /// `true`.
134 #[serde(default = "QuicSettings::default_enable_hystart")]
135 pub enable_hystart: bool,
136
137 /// Optionally enables pacing for outgoing packets.
138 ///
139 /// Note: this also requires pacing-compatible
140 /// [`SocketCapabilities`](crate::socket::SocketCapabilities).
141 pub enable_pacing: bool,
142
143 /// Optionally enables expensive versions of the
144 /// `accepted_initial_quic_packet_count`
145 /// and `rejected_initial_quic_packet_count` metrics.
146 ///
147 /// The expensive versions add a label for the peer IP subnet (`/24` for
148 /// IPv4, `/32` for IPv6). They thus generate many more time series if
149 /// peers are arbitrary eyeballs from the global Internet.
150 pub enable_expensive_packet_count_metrics: bool,
151
152 /// Forwards [`quiche`] logs into the logging system currently used by
153 /// [`foundations`]. Defaults to `false`.
154 ///
155 /// # Warning
156 /// This should **only be used for local debugging**. `quiche` can emit lots
157 /// (and lots, and lots) of logs (the TRACE level emits a log record for
158 /// every packet and frame) and you can very easily overwhelm your
159 /// logging pipeline.
160 pub capture_quiche_logs: bool,
161
162 /// A timeout for the QUIC handshake, in milliseconds. Disabled by default.
163 #[serde(rename = "handshake_timeout_ms")]
164 #[serde_as(as = "Option<DurationMilliSeconds>")]
165 pub handshake_timeout: Option<Duration>,
166
167 /// The maximum number of newly-created connections that will be queued for
168 /// the application to receive. Not applicable to client-side usage.
169 ///
170 /// Defaults to 1024 connections.
171 #[serde(default = "QuicSettings::default_listen_backlog")]
172 pub listen_backlog: usize,
173}
174
175impl QuicSettings {
176 #[inline]
177 fn default_alpn() -> Vec<Vec<u8>> {
178 quiche::h3::APPLICATION_PROTOCOL
179 .iter()
180 .map(|v| v.to_vec())
181 .collect()
182 }
183
184 #[inline]
185 fn default_enable_dgram() -> bool {
186 true
187 }
188
189 #[inline]
190 fn default_dgram_max_queue_len() -> usize {
191 65536
192 }
193
194 #[inline]
195 fn default_initial_max_data() -> u64 {
196 10_000_000
197 }
198
199 #[inline]
200 fn default_initial_max_stream_data() -> u64 {
201 1_000_000
202 }
203
204 #[inline]
205 fn default_initial_max_streams() -> u64 {
206 100
207 }
208
209 #[inline]
210 fn default_max_idle_timeout() -> Option<Duration> {
211 Some(Duration::from_secs(56))
212 }
213
214 #[inline]
215 fn default_max_recv_udp_payload_size() -> usize {
216 1350
217 }
218
219 #[inline]
220 fn default_max_send_udp_payload_size() -> usize {
221 1350
222 }
223
224 #[inline]
225 fn default_disable_active_migration() -> bool {
226 true
227 }
228
229 #[inline]
230 fn default_cc_algorithm() -> String {
231 "cubic".to_string()
232 }
233
234 #[inline]
235 fn default_enable_hystart() -> bool {
236 true
237 }
238
239 #[inline]
240 fn default_listen_backlog() -> usize {
241 // Given a worst-case 1 minute handshake timeout and up to 4096 concurrent
242 // handshakes, we will dequeue at least 70 connections per second.
243 // This means this backlog size limits the queueing latency to
244 // ~15s.
245 1024
246 }
247}
248
249#[cfg(test)]
250mod test {
251 use super::QuicSettings;
252 use std::time::Duration;
253
254 #[test]
255 fn timeouts_parse_as_milliseconds() {
256 let quic = serde_json::from_str::<QuicSettings>(
257 r#"{ "handshake_timeout_ms": 5000, "max_idle_timeout_ms": 7000 }"#,
258 )
259 .unwrap();
260
261 assert_eq!(quic.handshake_timeout.unwrap(), Duration::from_secs(5));
262 assert_eq!(quic.max_idle_timeout.unwrap(), Duration::from_secs(7));
263 }
264}