h3i/
lib.rs

1// Copyright (C) 2024, 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
27//! h3i - low-level HTTP/3 debug and testing
28//!
29//! HTTP/3 ([RFC 9114]) is the wire format for HTTP semantics ([RFC 9110]). The
30//! RFCs contain a range of requirements about how Request or Response messages
31//! are generated, serialized, sent, received, parsed, and consumed. QUIC ([RFC
32//! 9000]) streams are used for these messages along with other control and
33//! QPACK ([RFC 9204]) header compression instructions.
34//!
35//! h3i provides a highly configurable HTTP/3 client that can bend RFC rules in
36//! order to test the behavior of servers. QUIC streams can be opened, fin'd,
37//! stopped or reset at any point in time. HTTP/3 frames can be sent on any
38//! stream, in any order, containing user-controlled content (both legal and
39//! illegal).
40//!
41//! # Example
42//!
43//! The following example sends a request with its Content-Length header set to
44//! 5, but with its body only consisting of 4 bytes. This is classified as a
45//! [malformed request], and the server should respond with a 400 Bad Request
46//! response. Once h3i receives the response, it will close the connection.
47//!
48//! ```no_run
49//! use h3i::actions::h3::Action;
50//! use h3i::actions::h3::StreamEvent;
51//! use h3i::actions::h3::StreamEventType;
52//! use h3i::actions::h3::WaitType;
53//! use h3i::client::sync_client;
54//! use h3i::config::Config;
55//! use quiche::h3::frame::Frame;
56//! use quiche::h3::Header;
57//! use quiche::h3::NameValue;
58//!
59//! fn main() {
60//!    /// The QUIC stream to send the frames on. See
61//!    /// https://datatracker.ietf.org/doc/html/rfc9000#name-streams and
62//!    /// https://datatracker.ietf.org/doc/html/rfc9114#request-streams for more.
63//!    const STREAM_ID: u64 = 0;
64//!
65//!    let config = Config::new()
66//!        .with_host_port("blog.cloudflare.com".to_string())
67//!        .with_idle_timeout(2000)
68//!        .build()
69//!        .unwrap();
70//!
71//!    let headers = vec![
72//!        Header::new(b":method", b"POST"),
73//!        Header::new(b":scheme", b"https"),
74//!        Header::new(b":authority", b"blog.cloudflare.com"),
75//!        Header::new(b":path", b"/"),
76//!        // We say that we're going to send a body with 5 bytes...
77//!        Header::new(b"content-length", b"5"),
78//!    ];
79//!
80//!    let header_block = encode_header_block(&headers).unwrap();
81//!
82//!    let actions = vec![
83//!        Action::SendHeadersFrame {
84//!            stream_id: STREAM_ID,
85//!            fin_stream: false,
86//!            headers,
87//!            frame: Frame::Headers { header_block },
88//!        },
89//!        Action::SendFrame {
90//!            stream_id: STREAM_ID,
91//!            fin_stream: true,
92//!            frame: Frame::Data {
93//!                // ...but, in actuality, we only send 4 bytes. This should yield a
94//!                // 400 Bad Request response from an RFC-compliant
95//!                // server: https://datatracker.ietf.org/doc/html/rfc9114#section-4.1.2-3
96//!                payload: b"test".to_vec(),
97//!            },
98//!        },
99//!        Action::Wait {
100//!            wait_type: WaitType::StreamEvent(StreamEvent {
101//!                stream_id: STREAM_ID,
102//!                event_type: StreamEventType::Headers,
103//!            }),
104//!        },
105//!        Action::ConnectionClose {
106//!            error: quiche::ConnectionError {
107//!                is_app: true,
108//!                error_code: quiche::h3::WireErrorCode::NoError as u64,
109//!                reason: vec![],
110//!            },
111//!        },
112//!    ];
113//!
114//!    // This example doesn't use close trigger frames, since we manually close the connection upon
115//!    // receiving a HEADERS frame on stream 0.
116//!    let close_trigger_frames = None;
117//!    let summary = sync_client::connect(config, &actions, close_trigger_frames);
118//!
119//!    println!(
120//!        "=== received connection summary! ===\n\n{}",
121//!        serde_json::to_string_pretty(&summary).unwrap_or_else(|e| e.to_string())
122//!    );
123//! }
124//!
125//! // SendHeadersFrame requires a QPACK-encoded header block. h3i provides a
126//! // `send_headers_frame` helper function to abstract this, but for clarity, we do
127//! // it here.
128//! fn encode_header_block(
129//!     headers: &[quiche::h3::Header],
130//! ) -> std::result::Result<Vec<u8>, String> {
131//!     let mut encoder = quiche::h3::qpack::Encoder::new();
132//!
133//!     let headers_len = headers
134//!         .iter()
135//!         .fold(0, |acc, h| acc + h.value().len() + h.name().len() + 32);
136//!
137//!     let mut header_block = vec![0; headers_len];
138//!     let len = encoder
139//!         .encode(headers, &mut header_block)
140//!         .map_err(|_| "Internal Error")?;
141//!
142//!     header_block.truncate(len);
143//!
144//!     Ok(header_block)
145//! }
146//! ```
147
148//! [RFC 9000]: https://www.rfc-editor.org/rfc/rfc9000.html
149//! [RFC 9110]: https://www.rfc-editor.org/rfc/rfc9110.html
150//! [RFC 9114]: https://www.rfc-editor.org/rfc/rfc9114.html
151//! [RFC 9204]: https://www.rfc-editor.org/rfc/rfc9204.html
152//! [malformed request]: https://datatracker.ietf.org/doc/html/rfc9114#section-4.1.2-3
153
154use qlog::events::quic::PacketHeader;
155use qlog::events::quic::PacketSent;
156use qlog::events::quic::PacketType;
157use qlog::events::quic::QuicFrame;
158use qlog::events::EventData;
159pub use quiche;
160use quiche::h3::NameValue;
161
162use smallvec::SmallVec;
163
164/// The ID for an HTTP/3 control stream type.
165///
166/// See <https://datatracker.ietf.org/doc/html/rfc9114#name-control-streams>.
167pub const HTTP3_CONTROL_STREAM_TYPE_ID: u64 = 0x0;
168
169/// The ID for an HTTP/3 push stream type.
170///
171/// See <https://datatracker.ietf.org/doc/html/rfc9114#name-push-streams>.
172pub const HTTP3_PUSH_STREAM_TYPE_ID: u64 = 0x1;
173
174/// The ID for a QPACK encoder stream type.
175///
176/// See <https://datatracker.ietf.org/doc/html/rfc9204#section-4.2-2.1>.
177pub const QPACK_ENCODER_STREAM_TYPE_ID: u64 = 0x2;
178
179/// The ID for a QPACK decoder stream type.
180///
181/// See <https://datatracker.ietf.org/doc/html/rfc9204#section-4.2-2.2>.
182pub const QPACK_DECODER_STREAM_TYPE_ID: u64 = 0x3;
183
184#[derive(Default)]
185struct StreamIdAllocator {
186    id: u64,
187}
188
189impl StreamIdAllocator {
190    pub fn take_next_id(&mut self) -> u64 {
191        let old = self.id;
192        self.id += 4;
193
194        old
195    }
196
197    pub fn peek_next_id(&mut self) -> u64 {
198        self.id
199    }
200}
201
202fn encode_header_block(
203    headers: &[quiche::h3::Header],
204) -> std::result::Result<Vec<u8>, String> {
205    let mut encoder = quiche::h3::qpack::Encoder::new();
206
207    let headers_len = headers
208        .iter()
209        .fold(0, |acc, h| acc + h.value().len() + h.name().len() + 32);
210
211    let mut header_block = vec![0; headers_len];
212    let len = encoder
213        .encode(headers, &mut header_block)
214        .map_err(|_| "Internal Error")?;
215
216    header_block.truncate(len);
217
218    Ok(header_block)
219}
220
221fn fake_packet_header() -> PacketHeader {
222    PacketHeader {
223        packet_type: PacketType::OneRtt,
224        packet_number: None,
225        flags: None,
226        token: None,
227        length: None,
228        version: None,
229        scil: None,
230        dcil: None,
231        scid: None,
232        dcid: None,
233    }
234}
235
236fn fake_packet_sent(frames: Option<SmallVec<[QuicFrame; 1]>>) -> EventData {
237    EventData::PacketSent(PacketSent {
238        header: fake_packet_header(),
239        frames,
240        ..Default::default()
241    })
242}
243
244pub mod actions;
245pub mod client;
246pub mod config;
247pub mod frame;
248pub mod frame_parser;
249pub mod prompts;
250pub mod recordreplay;