| // Copyright 2013 The Chromium Authors |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "net/quic/quic_test_packet_maker.h" |
| |
| #include <list> |
| #include <utility> |
| |
| #include "base/strings/strcat.h" |
| #include "base/strings/string_number_conversions.h" |
| #include "net/base/features.h" |
| #include "net/quic/mock_crypto_client_stream.h" |
| #include "net/quic/quic_chromium_client_session.h" |
| #include "net/quic/quic_http_utils.h" |
| #include "net/spdy/spdy_http_utils.h" |
| #include "net/third_party/quiche/src/quiche/quic/core/http/http_constants.h" |
| #include "net/third_party/quiche/src/quiche/quic/core/quic_framer.h" |
| #include "net/third_party/quiche/src/quiche/quic/core/quic_stream.h" |
| #include "net/third_party/quiche/src/quiche/quic/core/quic_utils.h" |
| #include "net/third_party/quiche/src/quiche/quic/test_tools/mock_random.h" |
| #include "net/third_party/quiche/src/quiche/quic/test_tools/quic_test_utils.h" |
| |
| namespace net::test { |
| namespace { |
| |
| quic::QuicFrames CloneFrames(const quic::QuicFrames& frames) { |
| quic::QuicFrames new_frames = frames; |
| for (auto& frame : new_frames) { |
| switch (frame.type) { |
| // Frames smaller than a pointer are inlined, so don't need to be cloned. |
| case quic::PADDING_FRAME: |
| case quic::MTU_DISCOVERY_FRAME: |
| case quic::PING_FRAME: |
| case quic::MAX_STREAMS_FRAME: |
| case quic::STOP_WAITING_FRAME: |
| case quic::STREAMS_BLOCKED_FRAME: |
| case quic::STREAM_FRAME: |
| case quic::HANDSHAKE_DONE_FRAME: |
| case quic::BLOCKED_FRAME: |
| case quic::WINDOW_UPDATE_FRAME: |
| case quic::STOP_SENDING_FRAME: |
| case quic::PATH_CHALLENGE_FRAME: |
| case quic::PATH_RESPONSE_FRAME: |
| break; |
| case quic::ACK_FRAME: |
| frame.ack_frame = new quic::QuicAckFrame(*frame.ack_frame); |
| break; |
| case quic::RST_STREAM_FRAME: |
| frame.rst_stream_frame = |
| new quic::QuicRstStreamFrame(*frame.rst_stream_frame); |
| break; |
| case quic::CONNECTION_CLOSE_FRAME: |
| frame.connection_close_frame = |
| new quic::QuicConnectionCloseFrame(*frame.connection_close_frame); |
| break; |
| case quic::GOAWAY_FRAME: |
| frame.goaway_frame = new quic::QuicGoAwayFrame(*frame.goaway_frame); |
| break; |
| case quic::NEW_CONNECTION_ID_FRAME: |
| frame.new_connection_id_frame = |
| new quic::QuicNewConnectionIdFrame(*frame.new_connection_id_frame); |
| break; |
| case quic::RETIRE_CONNECTION_ID_FRAME: |
| frame.retire_connection_id_frame = |
| new quic::QuicRetireConnectionIdFrame( |
| *frame.retire_connection_id_frame); |
| break; |
| case quic::MESSAGE_FRAME: |
| DCHECK(false) << "Message frame not supported"; |
| // frame.message_frame = new |
| // quic::QuicMessageFrame(*frame.message_frame); |
| break; |
| case quic::CRYPTO_FRAME: |
| frame.crypto_frame = new quic::QuicCryptoFrame(*frame.crypto_frame); |
| break; |
| case quic::NEW_TOKEN_FRAME: |
| frame.new_token_frame = |
| new quic::QuicNewTokenFrame(*frame.new_token_frame); |
| break; |
| case quic::ACK_FREQUENCY_FRAME: |
| frame.ack_frequency_frame = |
| new quic::QuicAckFrequencyFrame(*frame.ack_frequency_frame); |
| break; |
| |
| case quic::NUM_FRAME_TYPES: |
| DCHECK(false) << "Cannot clone frame type: " << frame.type; |
| } |
| } |
| return new_frames; |
| } |
| |
| |
| } // namespace |
| |
| QuicTestPacketMaker::QuicTestPacketMaker(quic::ParsedQuicVersion version, |
| quic::QuicConnectionId connection_id, |
| const quic::QuicClock* clock, |
| const std::string& host, |
| quic::Perspective perspective, |
| bool client_priority_uses_incremental, |
| bool use_priority_header) |
| : version_(version), |
| connection_id_(connection_id), |
| clock_(clock), |
| host_(host), |
| qpack_encoder_(&decoder_stream_error_delegate_), |
| perspective_(perspective), |
| client_priority_uses_incremental_(client_priority_uses_incremental), |
| use_priority_header_(use_priority_header) { |
| DCHECK(!(perspective_ == quic::Perspective::IS_SERVER && |
| client_priority_uses_incremental_)); |
| |
| qpack_encoder_.set_qpack_stream_sender_delegate( |
| &encoder_stream_sender_delegate_); |
| } |
| |
| QuicTestPacketMaker::~QuicTestPacketMaker() { |
| for (auto& kv : saved_frames_) { |
| quic::DeleteFrames(&(kv.second)); |
| } |
| } |
| |
| void QuicTestPacketMaker::set_hostname(const std::string& host) { |
| host_.assign(host); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> |
| QuicTestPacketMaker::MakeConnectivityProbingPacket(uint64_t num) { |
| InitializeHeader(num); |
| |
| if (!version_.HasIetfQuicFrames()) { |
| AddQuicPingFrame(); |
| } else if (perspective_ == quic::Perspective::IS_CLIENT) { |
| AddQuicPathChallengeFrame(); |
| } else { |
| AddQuicPathResponseFrame(); |
| } |
| |
| AddQuicPaddingFrame(); |
| |
| return BuildPacket(); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> QuicTestPacketMaker::MakePingPacket( |
| uint64_t num) { |
| InitializeHeader(num); |
| AddQuicPingFrame(); |
| return BuildPacket(); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> |
| QuicTestPacketMaker::MakeRetireConnectionIdPacket(uint64_t num, |
| uint64_t sequence_number) { |
| InitializeHeader(num); |
| AddQuicRetireConnectionIdFrame(sequence_number); |
| return BuildPacket(); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> |
| QuicTestPacketMaker::MakeNewConnectionIdPacket( |
| uint64_t num, |
| const quic::QuicConnectionId& cid, |
| uint64_t sequence_number, |
| uint64_t retire_prior_to) { |
| InitializeHeader(num); |
| AddQuicNewConnectionIdFrame( |
| cid, sequence_number, retire_prior_to, |
| quic::QuicUtils::GenerateStatelessResetToken(cid)); |
| return BuildPacket(); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> |
| QuicTestPacketMaker::MakeAckAndNewConnectionIdPacket( |
| uint64_t num, |
| uint64_t largest_received, |
| uint64_t smallest_received, |
| const quic::QuicConnectionId& cid, |
| uint64_t sequence_number, |
| uint64_t retire_prior_to) { |
| InitializeHeader(num); |
| AddQuicAckFrame(largest_received, smallest_received); |
| AddQuicNewConnectionIdFrame( |
| cid, sequence_number, retire_prior_to, |
| quic::QuicUtils::GenerateStatelessResetToken(cid)); |
| return BuildPacket(); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> |
| QuicTestPacketMaker::MakeDummyCHLOPacket(uint64_t packet_num) { |
| SetEncryptionLevel(quic::ENCRYPTION_INITIAL); |
| InitializeHeader(packet_num); |
| |
| quic::CryptoHandshakeMessage message = |
| MockCryptoClientStream::GetDummyCHLOMessage(); |
| const quic::QuicData& data = message.GetSerialized(); |
| |
| if (!QuicVersionUsesCryptoFrames(version_.transport_version)) { |
| AddQuicStreamFrameWithOffset( |
| quic::QuicUtils::GetCryptoStreamId(version_.transport_version), |
| /*fin=*/false, /*offset=*/0, data.AsStringPiece()); |
| } else { |
| AddQuicCryptoFrame(quic::ENCRYPTION_INITIAL, 0, data.length()); |
| |
| data_producer_ = std::make_unique<quic::test::SimpleDataProducer>(); |
| data_producer_->SaveCryptoData(quic::ENCRYPTION_INITIAL, 0, |
| data.AsStringPiece()); |
| } |
| AddQuicPaddingFrame(); |
| |
| return BuildPacket(); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> |
| QuicTestPacketMaker::MakeAckAndPingPacket(uint64_t num, |
| uint64_t largest_received, |
| uint64_t smallest_received) { |
| InitializeHeader(num); |
| AddQuicAckFrame(largest_received, smallest_received); |
| AddQuicPingFrame(); |
| return BuildPacket(); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> |
| QuicTestPacketMaker::MakeAckAndRetireConnectionIdPacket( |
| uint64_t num, |
| uint64_t largest_received, |
| uint64_t smallest_received, |
| uint64_t sequence_number) { |
| InitializeHeader(num); |
| AddQuicAckFrame(largest_received, smallest_received); |
| AddQuicRetireConnectionIdFrame(sequence_number); |
| return BuildPacket(); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> |
| QuicTestPacketMaker::MakeRetransmissionAndRetireConnectionIdPacket( |
| uint64_t num, |
| const std::vector<uint64_t>& original_packet_numbers, |
| uint64_t sequence_number) { |
| InitializeHeader(num); |
| for (auto it : original_packet_numbers) { |
| for (auto frame : saved_frames_[quic::QuicPacketNumber(it)]) { |
| if (!MaybeCoalesceStreamFrame(frame)) { |
| frames_.push_back(frame); |
| } |
| } |
| } |
| AddQuicRetireConnectionIdFrame(sequence_number); |
| return BuildPacket(); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> |
| QuicTestPacketMaker::MakeStreamsBlockedPacket( |
| uint64_t num, |
| quic::QuicStreamCount stream_count, |
| bool unidirectional) { |
| InitializeHeader(num); |
| AddQuicStreamsBlockedFrame(1, stream_count, unidirectional); |
| return BuildPacket(); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> |
| QuicTestPacketMaker::MakeMaxStreamsPacket(uint64_t num, |
| quic::QuicStreamCount stream_count, |
| bool unidirectional) { |
| InitializeHeader(num); |
| AddQuicMaxStreamsFrame(1, stream_count, unidirectional); |
| return BuildPacket(); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> QuicTestPacketMaker::MakeRstPacket( |
| uint64_t num, |
| quic::QuicStreamId stream_id, |
| quic::QuicRstStreamErrorCode error_code) { |
| return MakeRstPacket(num, stream_id, error_code, |
| /*include_stop_sending_if_v99=*/true); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> QuicTestPacketMaker::MakeRstPacket( |
| uint64_t num, |
| quic::QuicStreamId stream_id, |
| quic::QuicRstStreamErrorCode error_code, |
| bool include_stop_sending_if_v99) { |
| InitializeHeader(num); |
| |
| if (include_stop_sending_if_v99 && version_.HasIetfQuicFrames()) { |
| AddQuicStopSendingFrame(stream_id, error_code); |
| } |
| if (!version_.HasIetfQuicFrames() || |
| quic::QuicUtils::IsBidirectionalStreamId(stream_id, version_)) { |
| AddQuicRstStreamFrame(stream_id, error_code); |
| } |
| |
| return BuildPacket(); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> |
| QuicTestPacketMaker::MakeRstAndDataPacket( |
| uint64_t num, |
| quic::QuicStreamId rst_stream_id, |
| quic::QuicRstStreamErrorCode rst_error_code, |
| quic::QuicStreamId data_stream_id, |
| std::string_view data) { |
| InitializeHeader(num); |
| |
| if (version_.HasIetfQuicFrames()) { |
| AddQuicStopSendingFrame(rst_stream_id, rst_error_code); |
| } |
| AddQuicRstStreamFrame(rst_stream_id, rst_error_code); |
| |
| AddQuicStreamFrame(data_stream_id, /* fin = */ false, data); |
| |
| return BuildPacket(); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> |
| QuicTestPacketMaker::MakeRetransmissionRstAndDataPacket( |
| const std::vector<uint64_t>& original_packet_numbers, |
| uint64_t num, |
| quic::QuicStreamId rst_stream_id, |
| quic::QuicRstStreamErrorCode rst_error_code, |
| quic::QuicStreamId data_stream_id, |
| std::string_view data, |
| uint64_t retransmit_frame_count) { |
| DCHECK(save_packet_frames_); |
| InitializeHeader(num); |
| |
| uint64_t frame_count = 0; |
| for (auto it : original_packet_numbers) { |
| for (auto frame : saved_frames_[quic::QuicPacketNumber(it)]) { |
| frame_count++; |
| if (retransmit_frame_count == 0 || |
| frame_count <= retransmit_frame_count) { |
| if (!MaybeCoalesceStreamFrame(frame)) { |
| frames_.push_back(frame); |
| } |
| } |
| } |
| } |
| |
| if (version_.HasIetfQuicFrames()) { |
| AddQuicStopSendingFrame(rst_stream_id, rst_error_code); |
| } |
| AddQuicRstStreamFrame(rst_stream_id, rst_error_code); |
| |
| AddQuicStreamFrame(data_stream_id, /* fin = */ false, data); |
| |
| return BuildPacket(); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> |
| QuicTestPacketMaker::MakeDataAndRstPacket( |
| uint64_t num, |
| quic::QuicStreamId data_stream_id, |
| std::string_view data, |
| quic::QuicStreamId rst_stream_id, |
| quic::QuicRstStreamErrorCode rst_error_code) { |
| InitializeHeader(num); |
| |
| AddQuicStreamFrame(data_stream_id, /* fin = */ false, data); |
| if (version_.HasIetfQuicFrames()) { |
| AddQuicStopSendingFrame(rst_stream_id, rst_error_code); |
| } |
| AddQuicRstStreamFrame(rst_stream_id, rst_error_code); |
| |
| return BuildPacket(); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> |
| QuicTestPacketMaker::MakeDataRstAndAckPacket( |
| uint64_t num, |
| quic::QuicStreamId data_stream_id, |
| std::string_view data, |
| quic::QuicStreamId rst_stream_id, |
| quic::QuicRstStreamErrorCode rst_error_code, |
| uint64_t largest_received, |
| uint64_t smallest_received) { |
| InitializeHeader(num); |
| |
| AddQuicAckFrame(largest_received, smallest_received); |
| |
| AddQuicStreamFrame(data_stream_id, /* fin = */ false, data); |
| if (version_.HasIetfQuicFrames()) { |
| AddQuicStopSendingFrame(rst_stream_id, rst_error_code); |
| } |
| AddQuicRstStreamFrame(rst_stream_id, rst_error_code); |
| |
| return BuildPacket(); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> |
| QuicTestPacketMaker::MakeAckAndRstPacket( |
| uint64_t num, |
| quic::QuicStreamId stream_id, |
| quic::QuicRstStreamErrorCode error_code, |
| uint64_t largest_received, |
| uint64_t smallest_received) { |
| return MakeAckAndRstPacket(num, stream_id, error_code, largest_received, |
| smallest_received, |
| /*include_stop_sending_if_v99=*/true); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> |
| QuicTestPacketMaker::MakeAckAndRstPacket( |
| uint64_t num, |
| quic::QuicStreamId stream_id, |
| quic::QuicRstStreamErrorCode error_code, |
| uint64_t largest_received, |
| uint64_t smallest_received, |
| bool include_stop_sending_if_v99) { |
| InitializeHeader(num); |
| |
| AddQuicAckFrame(largest_received, smallest_received); |
| |
| if (version_.HasIetfQuicFrames() && include_stop_sending_if_v99) { |
| AddQuicStopSendingFrame(stream_id, error_code); |
| } |
| if (!version_.HasIetfQuicFrames() || |
| quic::QuicUtils::IsBidirectionalStreamId(stream_id, version_)) { |
| AddQuicRstStreamFrame(stream_id, error_code); |
| } |
| |
| return BuildPacket(); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> |
| QuicTestPacketMaker::MakeRstAckAndConnectionClosePacket( |
| uint64_t num, |
| quic::QuicStreamId stream_id, |
| quic::QuicRstStreamErrorCode error_code, |
| uint64_t largest_received, |
| uint64_t smallest_received, |
| quic::QuicErrorCode quic_error, |
| const std::string& quic_error_details) { |
| InitializeHeader(num); |
| |
| AddQuicAckFrame(largest_received, smallest_received); |
| |
| if (version_.HasIetfQuicFrames()) { |
| AddQuicStopSendingFrame(stream_id, error_code); |
| } |
| AddQuicRstStreamFrame(stream_id, error_code); |
| AddQuicConnectionCloseFrame(quic_error, quic_error_details); |
| |
| return BuildPacket(); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> |
| QuicTestPacketMaker::MakeRstAckAndDataPacket( |
| uint64_t num, |
| quic::QuicStreamId stream_id, |
| quic::QuicRstStreamErrorCode error_code, |
| uint64_t largest_received, |
| uint64_t smallest_received, |
| quic::QuicStreamId data_id, |
| bool fin, |
| std::string_view data) { |
| InitializeHeader(num); |
| |
| AddQuicRstStreamFrame(stream_id, error_code); |
| |
| AddQuicAckFrame(largest_received, smallest_received); |
| AddQuicStreamFrame(data_id, fin, data); |
| return BuildPacket(); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> |
| QuicTestPacketMaker::MakeAckDataAndRst(uint64_t num, |
| quic::QuicStreamId stream_id, |
| quic::QuicRstStreamErrorCode error_code, |
| uint64_t largest_received, |
| uint64_t smallest_received, |
| quic::QuicStreamId data_id, |
| bool fin, |
| std::string_view data) { |
| InitializeHeader(num); |
| |
| AddQuicAckFrame(largest_received, smallest_received); |
| AddQuicStreamFrame(data_id, fin, data); |
| |
| AddQuicStopSendingFrame(stream_id, error_code); |
| AddQuicRstStreamFrame(stream_id, error_code); |
| |
| return BuildPacket(); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> |
| QuicTestPacketMaker::MakeAckRstAndDataPacket( |
| uint64_t num, |
| quic::QuicStreamId stream_id, |
| quic::QuicRstStreamErrorCode error_code, |
| uint64_t largest_received, |
| uint64_t smallest_received, |
| quic::QuicStreamId data_id, |
| bool fin, |
| std::string_view data) { |
| InitializeHeader(num); |
| |
| AddQuicAckFrame(largest_received, smallest_received); |
| AddQuicRstStreamFrame(stream_id, error_code); |
| AddQuicStreamFrame(data_id, fin, data); |
| return BuildPacket(); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> |
| QuicTestPacketMaker::MakeAckAndRetransmissionPacket( |
| uint64_t packet_number, |
| uint64_t first_received, |
| uint64_t largest_received, |
| uint64_t smallest_received, |
| const std::vector<uint64_t>& original_packet_numbers) { |
| DCHECK(save_packet_frames_); |
| InitializeHeader(packet_number); |
| AddQuicAckFrame(first_received, largest_received, smallest_received); |
| for (auto it : original_packet_numbers) { |
| for (auto frame : saved_frames_[quic::QuicPacketNumber(it)]) { |
| if (!MaybeCoalesceStreamFrame(frame)) { |
| frames_.push_back(frame); |
| } |
| } |
| } |
| |
| return BuildPacket(); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> |
| QuicTestPacketMaker::MakeCombinedRetransmissionPacket( |
| const std::vector<uint64_t>& original_packet_numbers, |
| uint64_t new_packet_number) { |
| DCHECK(save_packet_frames_); |
| InitializeHeader(new_packet_number); |
| for (auto it : original_packet_numbers) { |
| for (auto& frame : CloneFrames(saved_frames_[quic::QuicPacketNumber(it)])) { |
| if (frame.type != quic::PADDING_FRAME) { |
| if (!MaybeCoalesceStreamFrame(frame)) { |
| frames_.push_back(frame); |
| } |
| } |
| } |
| } |
| |
| return BuildPacket(); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> |
| QuicTestPacketMaker::MakeRstAndConnectionClosePacket( |
| uint64_t num, |
| quic::QuicStreamId stream_id, |
| quic::QuicRstStreamErrorCode error_code, |
| quic::QuicErrorCode quic_error, |
| const std::string& quic_error_details) { |
| InitializeHeader(num); |
| |
| if (version_.HasIetfQuicFrames()) { |
| AddQuicStopSendingFrame(stream_id, error_code); |
| } |
| AddQuicRstStreamFrame(stream_id, error_code); |
| |
| AddQuicConnectionCloseFrame(quic_error, quic_error_details); |
| |
| return BuildPacket(); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> |
| QuicTestPacketMaker::MakeDataRstAndConnectionClosePacket( |
| uint64_t num, |
| quic::QuicStreamId data_stream_id, |
| std::string_view data, |
| quic::QuicStreamId rst_stream_id, |
| quic::QuicRstStreamErrorCode error_code, |
| quic::QuicErrorCode quic_error, |
| const std::string& quic_error_details) { |
| InitializeHeader(num); |
| |
| AddQuicStreamFrame(data_stream_id, /* fin = */ false, data); |
| if (version_.HasIetfQuicFrames()) { |
| AddQuicStopSendingFrame(rst_stream_id, error_code); |
| } |
| AddQuicRstStreamFrame(rst_stream_id, error_code); |
| |
| AddQuicConnectionCloseFrame(quic_error, quic_error_details); |
| |
| return BuildPacket(); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> |
| QuicTestPacketMaker::MakeDataRstAckAndConnectionClosePacket( |
| uint64_t num, |
| quic::QuicStreamId data_stream_id, |
| std::string_view data, |
| quic::QuicStreamId rst_stream_id, |
| quic::QuicRstStreamErrorCode error_code, |
| uint64_t largest_received, |
| uint64_t smallest_received, |
| quic::QuicErrorCode quic_error, |
| const std::string& quic_error_details) { |
| InitializeHeader(num); |
| |
| AddQuicAckFrame(largest_received, smallest_received); |
| |
| AddQuicStreamFrame(data_stream_id, /* fin = */ false, data); |
| if (version_.HasIetfQuicFrames()) { |
| AddQuicStopSendingFrame(rst_stream_id, error_code); |
| } |
| AddQuicRstStreamFrame(rst_stream_id, error_code); |
| |
| AddQuicConnectionCloseFrame(quic_error, quic_error_details); |
| |
| return BuildPacket(); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> |
| QuicTestPacketMaker::MakeDataRstAckAndConnectionClosePacket( |
| uint64_t num, |
| quic::QuicStreamId data_stream_id, |
| std::string_view data, |
| quic::QuicStreamId rst_stream_id, |
| quic::QuicRstStreamErrorCode error_code, |
| uint64_t largest_received, |
| uint64_t smallest_received, |
| quic::QuicErrorCode quic_error, |
| const std::string& quic_error_details, |
| uint64_t frame_type) { |
| InitializeHeader(num); |
| |
| AddQuicStreamFrame(data_stream_id, /* fin = */ false, data); |
| if (version_.HasIetfQuicFrames()) { |
| AddQuicStopSendingFrame(rst_stream_id, error_code); |
| } |
| AddQuicRstStreamFrame(rst_stream_id, error_code); |
| |
| AddQuicAckFrame(largest_received, smallest_received); |
| AddQuicConnectionCloseFrame(quic_error, quic_error_details, frame_type); |
| |
| return BuildPacket(); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> |
| QuicTestPacketMaker::MakeStopSendingPacket( |
| uint64_t num, |
| quic::QuicStreamId stream_id, |
| quic::QuicRstStreamErrorCode error_code) { |
| DCHECK(version_.HasIetfQuicFrames()); |
| |
| InitializeHeader(num); |
| AddQuicStopSendingFrame(stream_id, error_code); |
| |
| return BuildPacket(); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> |
| QuicTestPacketMaker::MakeAckAndConnectionClosePacket( |
| uint64_t num, |
| uint64_t largest_received, |
| uint64_t smallest_received, |
| quic::QuicErrorCode quic_error, |
| const std::string& quic_error_details, |
| uint64_t frame_type) { |
| InitializeHeader(num); |
| AddQuicAckFrame(largest_received, smallest_received); |
| AddQuicConnectionCloseFrame(quic_error, quic_error_details, frame_type); |
| return BuildPacket(); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> |
| QuicTestPacketMaker::MakeConnectionClosePacket( |
| uint64_t num, |
| quic::QuicErrorCode quic_error, |
| const std::string& quic_error_details) { |
| InitializeHeader(num); |
| AddQuicConnectionCloseFrame(quic_error, quic_error_details); |
| return BuildPacket(); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> QuicTestPacketMaker::MakeGoAwayPacket( |
| uint64_t num, |
| quic::QuicErrorCode error_code, |
| std::string reason_phrase) { |
| InitializeHeader(num); |
| AddQuicGoAwayFrame(error_code, reason_phrase); |
| return BuildPacket(); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> QuicTestPacketMaker::MakeAckPacket( |
| uint64_t packet_number, |
| uint64_t largest_received, |
| uint64_t smallest_received) { |
| return MakeAckPacket(packet_number, 1, largest_received, smallest_received); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> QuicTestPacketMaker::MakeAckPacket( |
| uint64_t packet_number, |
| uint64_t first_received, |
| uint64_t largest_received, |
| uint64_t smallest_received) { |
| InitializeHeader(packet_number); |
| AddQuicAckFrame(first_received, largest_received, smallest_received); |
| return BuildPacket(); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> QuicTestPacketMaker::MakeDataPacket( |
| uint64_t packet_number, |
| quic::QuicStreamId stream_id, |
| bool fin, |
| std::string_view data) { |
| InitializeHeader(packet_number); |
| AddQuicStreamFrame(stream_id, fin, data); |
| return BuildPacket(); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> |
| QuicTestPacketMaker::MakeAckAndDataPacket(uint64_t packet_number, |
| quic::QuicStreamId stream_id, |
| uint64_t largest_received, |
| uint64_t smallest_received, |
| bool fin, |
| std::string_view data) { |
| InitializeHeader(packet_number); |
| |
| AddQuicAckFrame(largest_received, smallest_received); |
| AddQuicStreamFrame(stream_id, fin, data); |
| |
| return BuildPacket(); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> |
| QuicTestPacketMaker::MakeAckRetransmissionAndDataPacket( |
| uint64_t packet_number, |
| const std::vector<uint64_t>& original_packet_numbers, |
| quic::QuicStreamId stream_id, |
| uint64_t largest_received, |
| uint64_t smallest_received, |
| bool fin, |
| std::string_view data) { |
| InitializeHeader(packet_number); |
| |
| AddQuicAckFrame(largest_received, smallest_received); |
| for (auto it : original_packet_numbers) { |
| for (auto frame : saved_frames_[quic::QuicPacketNumber(it)]) { |
| if (!MaybeCoalesceStreamFrame(frame)) { |
| frames_.push_back(frame); |
| } |
| } |
| } |
| AddQuicStreamFrame(stream_id, fin, data); |
| |
| return BuildPacket(); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> |
| QuicTestPacketMaker::MakeRequestHeadersAndMultipleDataFramesPacket( |
| uint64_t packet_number, |
| quic::QuicStreamId stream_id, |
| bool fin, |
| spdy::SpdyPriority spdy_priority, |
| spdy::Http2HeaderBlock headers, |
| size_t* spdy_headers_frame_length, |
| const std::vector<std::string>& data_writes) { |
| InitializeHeader(packet_number); |
| |
| MaybeAddHttp3SettingsFrames(); |
| |
| std::string priority_data = |
| GenerateHttp3PriorityData(spdy_priority, stream_id); |
| if (!priority_data.empty()) { |
| AddQuicStreamFrame(2, false, priority_data); |
| } |
| |
| AddPriorityHeader(spdy_priority, &headers); |
| std::string data = QpackEncodeHeaders(stream_id, std::move(headers), |
| spdy_headers_frame_length); |
| for (const auto& data_write : data_writes) { |
| data += data_write; |
| } |
| AddQuicStreamFrame(stream_id, fin, data); |
| |
| return BuildPacket(); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> |
| QuicTestPacketMaker::MakeRequestHeadersPacket( |
| uint64_t packet_number, |
| quic::QuicStreamId stream_id, |
| bool fin, |
| spdy::SpdyPriority spdy_priority, |
| spdy::Http2HeaderBlock headers, |
| size_t* spdy_headers_frame_length, |
| bool should_include_priority_frame) { |
| InitializeHeader(packet_number); |
| |
| MaybeAddHttp3SettingsFrames(); |
| |
| if (should_include_priority_frame) { |
| std::string priority_data = |
| GenerateHttp3PriorityData(spdy_priority, stream_id); |
| if (!priority_data.empty()) { |
| AddQuicStreamFrame(2, false, priority_data); |
| } |
| } |
| |
| AddPriorityHeader(spdy_priority, &headers); |
| std::string data = QpackEncodeHeaders(stream_id, std::move(headers), |
| spdy_headers_frame_length); |
| AddQuicStreamFrame(stream_id, fin, data); |
| |
| return BuildPacket(); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> |
| QuicTestPacketMaker::MakeRetransmissionAndRequestHeadersPacket( |
| const std::vector<uint64_t>& original_packet_numbers, |
| uint64_t packet_number, |
| quic::QuicStreamId stream_id, |
| bool fin, |
| spdy::SpdyPriority spdy_priority, |
| spdy::Http2HeaderBlock headers, |
| size_t* spdy_headers_frame_length) { |
| DCHECK(save_packet_frames_); |
| InitializeHeader(packet_number); |
| |
| for (auto it : original_packet_numbers) { |
| for (auto frame : saved_frames_[quic::QuicPacketNumber(it)]) { |
| if (!MaybeCoalesceStreamFrame(frame)) { |
| frames_.push_back(frame); |
| } |
| } |
| } |
| |
| MaybeAddHttp3SettingsFrames(); |
| |
| std::string priority_data = |
| GenerateHttp3PriorityData(spdy_priority, stream_id); |
| if (!priority_data.empty()) { |
| AddQuicStreamFrame(2, false, priority_data); |
| } |
| |
| AddPriorityHeader(spdy_priority, &headers); |
| std::string data = QpackEncodeHeaders(stream_id, std::move(headers), |
| spdy_headers_frame_length); |
| AddQuicStreamFrame(stream_id, fin, data); |
| |
| return BuildPacket(); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> |
| QuicTestPacketMaker::MakeRequestHeadersAndRstPacket( |
| uint64_t packet_number, |
| quic::QuicStreamId stream_id, |
| bool fin, |
| spdy::SpdyPriority spdy_priority, |
| spdy::Http2HeaderBlock headers, |
| size_t* spdy_headers_frame_length, |
| quic::QuicRstStreamErrorCode error_code) { |
| InitializeHeader(packet_number); |
| |
| MaybeAddHttp3SettingsFrames(); |
| |
| std::string priority_data = |
| GenerateHttp3PriorityData(spdy_priority, stream_id); |
| if (!priority_data.empty()) { |
| AddQuicStreamFrame(2, false, priority_data); |
| } |
| |
| AddPriorityHeader(spdy_priority, &headers); |
| std::string data = QpackEncodeHeaders(stream_id, std::move(headers), |
| spdy_headers_frame_length); |
| AddQuicStreamFrame(stream_id, fin, data); |
| AddQuicStopSendingFrame(stream_id, error_code); |
| AddQuicRstStreamFrame(stream_id, error_code); |
| |
| return BuildPacket(); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> |
| QuicTestPacketMaker::MakeResponseHeadersPacket( |
| uint64_t packet_number, |
| quic::QuicStreamId stream_id, |
| bool fin, |
| spdy::Http2HeaderBlock headers, |
| size_t* spdy_headers_frame_length) { |
| InitializeHeader(packet_number); |
| |
| std::string data = QpackEncodeHeaders(stream_id, std::move(headers), |
| spdy_headers_frame_length); |
| |
| AddQuicStreamFrame(stream_id, fin, data); |
| |
| return BuildPacket(); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> |
| QuicTestPacketMaker::MakeInitialSettingsPacket(uint64_t packet_number) { |
| InitializeHeader(packet_number); |
| MaybeAddHttp3SettingsFrames(); |
| return BuildPacket(); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> |
| QuicTestPacketMaker::MakePriorityPacket(uint64_t packet_number, |
| quic::QuicStreamId id, |
| spdy::SpdyPriority spdy_priority) { |
| InitializeHeader(packet_number); |
| |
| std::string priority_data = GenerateHttp3PriorityData(spdy_priority, id); |
| if (!priority_data.empty()) { |
| AddQuicStreamFrame(2, false, priority_data); |
| } |
| |
| return BuildPacket(); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> |
| QuicTestPacketMaker::MakeAckAndPriorityPacket( |
| uint64_t packet_number, |
| uint64_t largest_received, |
| uint64_t smallest_received, |
| quic::QuicStreamId id, |
| spdy::SpdyPriority spdy_priority) { |
| InitializeHeader(packet_number); |
| |
| AddQuicAckFrame(largest_received, smallest_received); |
| |
| std::string priority_data = GenerateHttp3PriorityData(spdy_priority, id); |
| if (!priority_data.empty()) { |
| AddQuicStreamFrame(2, false, priority_data); |
| } |
| |
| return BuildPacket(); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> |
| QuicTestPacketMaker::MakeAckAndPriorityUpdatePacket( |
| uint64_t packet_number, |
| uint64_t largest_received, |
| uint64_t smallest_received, |
| quic::QuicStreamId id, |
| spdy::SpdyPriority spdy_priority) { |
| InitializeHeader(packet_number); |
| |
| AddQuicAckFrame(largest_received, smallest_received); |
| |
| std::string priority_data = GenerateHttp3PriorityData(spdy_priority, id); |
| if (!priority_data.empty()) { |
| AddQuicStreamFrame(2, false, priority_data); |
| } |
| |
| return BuildPacket(); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> |
| QuicTestPacketMaker::MakeRetransmissionPacket(uint64_t original_packet_number, |
| uint64_t new_packet_number) { |
| DCHECK(save_packet_frames_); |
| InitializeHeader(new_packet_number); |
| return BuildPacketImpl( |
| saved_frames_[quic::QuicPacketNumber(original_packet_number)], nullptr); |
| } |
| |
| std::unique_ptr<quic::QuicEncryptedPacket> |
| QuicTestPacketMaker::MakeStatelessResetPacket() { |
| auto connection_id = quic::test::TestConnectionId(); |
| return quic::QuicFramer::BuildIetfStatelessResetPacket( |
| connection_id, quic::QuicFramer::GetMinStatelessResetPacketLength() + 1, |
| quic::QuicUtils::GenerateStatelessResetToken(connection_id)); |
| } |
| |
| void QuicTestPacketMaker::RemoveSavedStreamFrames( |
| quic::QuicStreamId stream_id) { |
| for (auto& kv : saved_frames_) { |
| auto* it = kv.second.begin(); |
| while (it != kv.second.end()) { |
| if (it->type == quic::STREAM_FRAME && |
| it->stream_frame.stream_id == stream_id) { |
| it = kv.second.erase(it); |
| } else { |
| ++it; |
| } |
| } |
| } |
| } |
| |
| void QuicTestPacketMaker::SetEncryptionLevel(quic::EncryptionLevel level) { |
| encryption_level_ = level; |
| switch (level) { |
| case quic::ENCRYPTION_INITIAL: |
| long_header_type_ = quic::INITIAL; |
| break; |
| case quic::ENCRYPTION_ZERO_RTT: |
| long_header_type_ = quic::ZERO_RTT_PROTECTED; |
| break; |
| case quic::ENCRYPTION_FORWARD_SECURE: |
| long_header_type_ = quic::INVALID_PACKET_TYPE; |
| break; |
| default: |
| LOG(DFATAL) << quic::EncryptionLevelToString(level); |
| long_header_type_ = quic::INVALID_PACKET_TYPE; |
| } |
| } |
| |
| spdy::Http2HeaderBlock QuicTestPacketMaker::GetRequestHeaders( |
| const std::string& method, |
| const std::string& scheme, |
| const std::string& path) const { |
| spdy::Http2HeaderBlock headers; |
| headers[":method"] = method; |
| headers[":authority"] = host_; |
| headers[":scheme"] = scheme; |
| headers[":path"] = path; |
| return headers; |
| } |
| |
| spdy::Http2HeaderBlock QuicTestPacketMaker::ConnectRequestHeaders( |
| const std::string& host_port) const { |
| spdy::Http2HeaderBlock headers; |
| headers[":method"] = "CONNECT"; |
| headers[":authority"] = host_port; |
| return headers; |
| } |
| |
| spdy::Http2HeaderBlock QuicTestPacketMaker::GetResponseHeaders( |
| const std::string& status) const { |
| spdy::Http2HeaderBlock headers; |
| headers[":status"] = status; |
| headers["content-type"] = "text/plain"; |
| return headers; |
| } |
| |
| spdy::Http2HeaderBlock QuicTestPacketMaker::GetResponseHeaders( |
| const std::string& status, |
| const std::string& alt_svc) const { |
| spdy::Http2HeaderBlock headers; |
| headers[":status"] = status; |
| headers["alt-svc"] = alt_svc; |
| headers["content-type"] = "text/plain"; |
| return headers; |
| } |
| |
| void QuicTestPacketMaker::Reset() { |
| stream_offsets_.clear(); |
| } |
| |
| std::string QuicTestPacketMaker::QpackEncodeHeaders( |
| quic::QuicStreamId stream_id, |
| spdy::Http2HeaderBlock headers, |
| size_t* encoded_data_length) { |
| std::string data; |
| |
| std::string encoded_headers = |
| qpack_encoder_.EncodeHeaderList(stream_id, headers, nullptr); |
| |
| // Generate HEADERS frame header. |
| const std::string headers_frame_header = |
| quic::HttpEncoder::SerializeHeadersFrameHeader(encoded_headers.size()); |
| |
| // Add the HEADERS frame header. |
| data += headers_frame_header; |
| // Add the HEADERS frame payload. |
| data += encoded_headers; |
| |
| // Compute the total data length. |
| if (encoded_data_length) { |
| *encoded_data_length = data.length(); |
| } |
| return data; |
| } |
| |
| void QuicTestPacketMaker::InitializeHeader(uint64_t packet_number) { |
| header_.destination_connection_id = DestinationConnectionId(); |
| header_.destination_connection_id_included = HasDestinationConnectionId(); |
| header_.source_connection_id = SourceConnectionId(); |
| header_.source_connection_id_included = HasSourceConnectionId(); |
| header_.reset_flag = false; |
| header_.version_flag = ShouldIncludeVersion(); |
| header_.form = header_.version_flag ? quic::IETF_QUIC_LONG_HEADER_PACKET |
| : quic::IETF_QUIC_SHORT_HEADER_PACKET; |
| header_.long_packet_type = long_header_type_; |
| header_.packet_number_length = GetPacketNumberLength(); |
| header_.packet_number = quic::QuicPacketNumber(packet_number); |
| if (quic::QuicVersionHasLongHeaderLengths(version_.transport_version) && |
| header_.version_flag) { |
| if (long_header_type_ == quic::INITIAL) { |
| header_.retry_token_length_length = |
| quiche::VARIABLE_LENGTH_INTEGER_LENGTH_1; |
| } |
| header_.length_length = quiche::VARIABLE_LENGTH_INTEGER_LENGTH_2; |
| } |
| } |
| |
| void QuicTestPacketMaker::AddQuicPaddingFrame() { |
| quic::QuicPaddingFrame padding_frame; |
| frames_.push_back(quic::QuicFrame(padding_frame)); |
| DVLOG(1) << "Adding frame: " << frames_.back(); |
| } |
| |
| void QuicTestPacketMaker::AddQuicPingFrame() { |
| quic::QuicPingFrame ping_frame; |
| frames_.push_back(quic::QuicFrame(ping_frame)); |
| DVLOG(1) << "Adding frame: " << frames_.back(); |
| } |
| |
| void QuicTestPacketMaker::AddQuicRetireConnectionIdFrame( |
| uint64_t sequence_number) { |
| auto* retire_cid_frame = new quic::QuicRetireConnectionIdFrame(); |
| retire_cid_frame->sequence_number = sequence_number; |
| frames_.push_back(quic::QuicFrame(retire_cid_frame)); |
| DVLOG(1) << "Adding frame: " << frames_.back(); |
| } |
| |
| void QuicTestPacketMaker::AddQuicNewConnectionIdFrame( |
| const quic::QuicConnectionId& cid, |
| uint64_t sequence_number, |
| uint64_t retire_prior_to, |
| quic::StatelessResetToken reset_token) { |
| auto* new_cid_frame = new quic::QuicNewConnectionIdFrame(); |
| new_cid_frame->connection_id = cid; |
| new_cid_frame->sequence_number = sequence_number; |
| new_cid_frame->retire_prior_to = retire_prior_to; |
| new_cid_frame->stateless_reset_token = reset_token; |
| frames_.push_back(quic::QuicFrame(new_cid_frame)); |
| DVLOG(1) << "Adding frame: " << frames_.back(); |
| } |
| |
| void QuicTestPacketMaker::AddQuicMaxStreamsFrame( |
| quic::QuicControlFrameId control_frame_id, |
| quic::QuicStreamCount stream_count, |
| bool unidirectional) { |
| quic::QuicMaxStreamsFrame max_streams_frame(control_frame_id, stream_count, |
| unidirectional); |
| frames_.push_back(quic::QuicFrame(max_streams_frame)); |
| DVLOG(1) << "Adding frame: " << frames_.back(); |
| } |
| |
| void QuicTestPacketMaker::AddQuicStreamsBlockedFrame( |
| quic::QuicControlFrameId control_frame_id, |
| quic::QuicStreamCount stream_count, |
| bool unidirectional) { |
| quic::QuicStreamsBlockedFrame streams_blocked_frame( |
| control_frame_id, stream_count, unidirectional); |
| frames_.push_back(quic::QuicFrame(streams_blocked_frame)); |
| DVLOG(1) << "Adding frame: " << frames_.back(); |
| } |
| |
| void QuicTestPacketMaker::AddQuicStreamFrame(quic::QuicStreamId stream_id, |
| bool fin, |
| std::string_view data) { |
| AddQuicStreamFrameWithOffset(stream_id, fin, stream_offsets_[stream_id], |
| data); |
| stream_offsets_[stream_id] += data.length(); |
| } |
| |
| void QuicTestPacketMaker::AddQuicStreamFrameWithOffset( |
| quic::QuicStreamId stream_id, |
| bool fin, |
| quic::QuicStreamOffset offset, |
| std::string_view data) { |
| // Save the stream data so that callers can use temporary objects for data. |
| saved_stream_data_.push_back(std::make_unique<std::string>(data)); |
| std::string_view saved_data = *saved_stream_data_.back(); |
| |
| quic::QuicStreamFrame stream_frame(stream_id, fin, offset, saved_data); |
| frames_.push_back(quic::QuicFrame(stream_frame)); |
| DVLOG(1) << "Adding frame: " << frames_.back(); |
| } |
| |
| void QuicTestPacketMaker::AddQuicAckFrame(uint64_t largest_received, |
| uint64_t smallest_received) { |
| AddQuicAckFrame(1, largest_received, smallest_received); |
| } |
| |
| void QuicTestPacketMaker::AddQuicAckFrame(uint64_t first_received, |
| uint64_t largest_received, |
| uint64_t smallest_received) { |
| auto* ack_frame = new quic::QuicAckFrame; |
| ack_frame->largest_acked = quic::QuicPacketNumber(largest_received); |
| ack_frame->ack_delay_time = quic::QuicTime::Delta::Zero(); |
| for (uint64_t i = smallest_received; i <= largest_received; ++i) { |
| ack_frame->received_packet_times.push_back( |
| std::make_pair(quic::QuicPacketNumber(i), clock_->Now())); |
| } |
| if (largest_received > 0) { |
| DCHECK_GE(largest_received, first_received); |
| ack_frame->packets.AddRange(quic::QuicPacketNumber(first_received), |
| quic::QuicPacketNumber(largest_received + 1)); |
| } |
| frames_.push_back(quic::QuicFrame(ack_frame)); |
| DVLOG(1) << "Adding frame: " << frames_.back(); |
| } |
| |
| void QuicTestPacketMaker::AddQuicRstStreamFrame( |
| quic::QuicStreamId stream_id, |
| quic::QuicRstStreamErrorCode error_code) { |
| auto* rst_stream_frame = new quic::QuicRstStreamFrame( |
| 1, stream_id, error_code, stream_offsets_[stream_id]); |
| frames_.push_back(quic::QuicFrame(rst_stream_frame)); |
| DVLOG(1) << "Adding frame: " << frames_.back(); |
| } |
| |
| void QuicTestPacketMaker::AddQuicConnectionCloseFrame( |
| quic::QuicErrorCode quic_error, |
| const std::string& quic_error_details) { |
| AddQuicConnectionCloseFrame(quic_error, quic_error_details, 0); |
| } |
| |
| void QuicTestPacketMaker::AddQuicConnectionCloseFrame( |
| quic::QuicErrorCode quic_error, |
| const std::string& quic_error_details, |
| uint64_t frame_type) { |
| auto* close_frame = new quic::QuicConnectionCloseFrame( |
| version_.transport_version, quic_error, quic::NO_IETF_QUIC_ERROR, |
| quic_error_details, frame_type); |
| frames_.push_back(quic::QuicFrame(close_frame)); |
| DVLOG(1) << "Adding frame: " << frames_.back(); |
| } |
| |
| void QuicTestPacketMaker::AddQuicGoAwayFrame(quic::QuicErrorCode error_code, |
| std::string reason_phrase) { |
| auto* goaway_frame = new quic::QuicGoAwayFrame(); |
| goaway_frame->error_code = error_code; |
| goaway_frame->last_good_stream_id = 0; |
| goaway_frame->reason_phrase = reason_phrase; |
| frames_.push_back(quic::QuicFrame(goaway_frame)); |
| DVLOG(1) << "Adding frame: " << frames_.back(); |
| } |
| |
| void QuicTestPacketMaker::AddQuicPathResponseFrame() { |
| quic::test::MockRandom rand(0); |
| quic::QuicPathFrameBuffer payload; |
| rand.RandBytes(payload.data(), payload.size()); |
| auto path_response_frame = quic::QuicPathResponseFrame(0, payload); |
| frames_.push_back(quic::QuicFrame(path_response_frame)); |
| DVLOG(1) << "Adding frame: " << frames_.back(); |
| } |
| |
| void QuicTestPacketMaker::AddQuicPathChallengeFrame() { |
| quic::test::MockRandom rand(0); |
| quic::QuicPathFrameBuffer payload; |
| rand.RandBytes(payload.data(), payload.size()); |
| auto path_challenge_frame = quic::QuicPathChallengeFrame(0, payload); |
| frames_.push_back(quic::QuicFrame(path_challenge_frame)); |
| DVLOG(1) << "Adding frame: " << frames_.back(); |
| } |
| |
| void QuicTestPacketMaker::AddQuicStopSendingFrame( |
| quic::QuicStreamId stream_id, |
| quic::QuicRstStreamErrorCode error_code) { |
| auto stop_sending_frame = |
| quic::QuicStopSendingFrame(1, stream_id, error_code); |
| frames_.push_back(quic::QuicFrame(stop_sending_frame)); |
| DVLOG(1) << "Adding frame: " << frames_.back(); |
| } |
| |
| void QuicTestPacketMaker::AddQuicCryptoFrame( |
| quic::EncryptionLevel level, |
| quic::QuicStreamOffset offset, |
| quic::QuicPacketLength data_length) { |
| auto* crypto_frame = new quic::QuicCryptoFrame(level, offset, data_length); |
| frames_.push_back(quic::QuicFrame(crypto_frame)); |
| DVLOG(1) << "Adding frame: " << frames_.back(); |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> QuicTestPacketMaker::BuildPacket() { |
| auto packet = BuildPacketImpl(frames_, data_producer_.get()); |
| |
| DeleteFrames(&frames_); |
| data_producer_.reset(); |
| |
| return packet; |
| } |
| |
| std::unique_ptr<quic::QuicReceivedPacket> QuicTestPacketMaker::BuildPacketImpl( |
| const quic::QuicFrames& frames, |
| quic::QuicStreamFrameDataProducer* data_producer) { |
| quic::QuicFramer framer(quic::test::SupportedVersions(version_), |
| clock_->Now(), perspective_, |
| quic::kQuicDefaultConnectionIdLength); |
| if (encryption_level_ == quic::ENCRYPTION_INITIAL) { |
| framer.SetInitialObfuscators(perspective_ == quic::Perspective::IS_CLIENT |
| ? header_.destination_connection_id |
| : header_.source_connection_id); |
| } else { |
| framer.SetEncrypter( |
| encryption_level_, |
| std::make_unique<quic::test::TaggingEncrypter>( // IN-TEST |
| encryption_level_)); |
| } |
| if (data_producer != nullptr) { |
| framer.set_data_producer(data_producer); |
| } |
| quic::QuicFrames frames_copy = CloneFrames(frames); |
| size_t max_plaintext_size = |
| framer.GetMaxPlaintextSize(quic::kDefaultMaxPacketSize); |
| if (version_.HasHeaderProtection()) { |
| size_t packet_size = |
| quic::GetPacketHeaderSize(version_.transport_version, header_); |
| size_t frames_size = 0; |
| for (size_t i = 0; i < frames.size(); ++i) { |
| bool first_frame = i == 0; |
| bool last_frame = i == frames.size() - 1; |
| const size_t frame_size = framer.GetSerializedFrameLength( |
| frames[i], max_plaintext_size - packet_size, first_frame, last_frame, |
| header_.packet_number_length); |
| packet_size += frame_size; |
| frames_size += frame_size; |
| } |
| |
| const size_t min_plaintext_packet_size = |
| quic::QuicPacketCreator::MinPlaintextPacketSize( |
| version_, header_.packet_number_length); |
| if (frames_size < min_plaintext_packet_size) { |
| frames_copy.insert(frames_copy.begin(), |
| quic::QuicFrame(quic::QuicPaddingFrame( |
| min_plaintext_packet_size - frames_size))); |
| } |
| } |
| std::unique_ptr<quic::QuicPacket> packet(quic::test::BuildUnsizedDataPacket( |
| &framer, header_, frames_copy, max_plaintext_size)); |
| char buffer[quic::kMaxOutgoingPacketSize]; |
| size_t encrypted_size = |
| framer.EncryptPayload(encryption_level_, header_.packet_number, *packet, |
| buffer, quic::kMaxOutgoingPacketSize); |
| EXPECT_NE(0u, encrypted_size); |
| quic::QuicReceivedPacket encrypted(buffer, encrypted_size, clock_->Now(), |
| false); |
| if (save_packet_frames_) { |
| saved_frames_[header_.packet_number] = frames_copy; |
| } else { |
| saved_stream_data_.clear(); |
| DeleteFrames(&frames_copy); |
| } |
| |
| return encrypted.Clone(); |
| } |
| |
| bool QuicTestPacketMaker::ShouldIncludeVersion() const { |
| return encryption_level_ < quic::ENCRYPTION_FORWARD_SECURE; |
| } |
| |
| quic::QuicPacketNumberLength QuicTestPacketMaker::GetPacketNumberLength() |
| const { |
| if (ShouldIncludeVersion() && |
| !version_.SendsVariableLengthPacketNumberInLongHeader()) { |
| return quic::PACKET_4BYTE_PACKET_NUMBER; |
| } |
| return quic::PACKET_1BYTE_PACKET_NUMBER; |
| } |
| |
| quic::QuicConnectionId QuicTestPacketMaker::DestinationConnectionId() const { |
| if (perspective_ == quic::Perspective::IS_SERVER) { |
| return quic::EmptyQuicConnectionId(); |
| } |
| return connection_id_; |
| } |
| |
| quic::QuicConnectionId QuicTestPacketMaker::SourceConnectionId() const { |
| if (perspective_ == quic::Perspective::IS_CLIENT) { |
| return quic::EmptyQuicConnectionId(); |
| } |
| return connection_id_; |
| } |
| |
| quic::QuicConnectionIdIncluded QuicTestPacketMaker::HasDestinationConnectionId() |
| const { |
| if (!version_.SupportsClientConnectionIds() && |
| perspective_ == quic::Perspective::IS_SERVER) { |
| return quic::CONNECTION_ID_ABSENT; |
| } |
| return quic::CONNECTION_ID_PRESENT; |
| } |
| |
| quic::QuicConnectionIdIncluded QuicTestPacketMaker::HasSourceConnectionId() |
| const { |
| if (version_.SupportsClientConnectionIds() || |
| (perspective_ == quic::Perspective::IS_SERVER && |
| encryption_level_ < quic::ENCRYPTION_FORWARD_SECURE)) { |
| return quic::CONNECTION_ID_PRESENT; |
| } |
| return quic::CONNECTION_ID_ABSENT; |
| } |
| |
| quic::QuicStreamId QuicTestPacketMaker::GetFirstBidirectionalStreamId() const { |
| return quic::QuicUtils::GetFirstBidirectionalStreamId( |
| version_.transport_version, perspective_); |
| } |
| |
| quic::QuicStreamId QuicTestPacketMaker::GetHeadersStreamId() const { |
| return quic::QuicUtils::GetHeadersStreamId(version_.transport_version); |
| } |
| |
| std::string QuicTestPacketMaker::GenerateHttp3SettingsData() { |
| quic::SettingsFrame settings; |
| settings.values[quic::SETTINGS_MAX_FIELD_SECTION_SIZE] = |
| kQuicMaxHeaderListSize; |
| settings.values[quic::SETTINGS_QPACK_MAX_TABLE_CAPACITY] = |
| quic::kDefaultQpackMaxDynamicTableCapacity; |
| settings.values[quic::SETTINGS_QPACK_BLOCKED_STREAMS] = |
| quic::kDefaultMaximumBlockedStreams; |
| if (FLAGS_quic_reloadable_flag_quic_enable_h3_datagrams) { |
| settings.values[quic::SETTINGS_H3_DATAGRAM] = 1; |
| } |
| // Greased setting. |
| settings.values[0x40] = 20; |
| return quic::HttpEncoder::SerializeSettingsFrame(settings); |
| } |
| |
| std::string QuicTestPacketMaker::GenerateHttp3PriorityData( |
| spdy::SpdyPriority spdy_priority, |
| quic::QuicStreamId stream_id) { |
| std::string priority_data; |
| quic::PriorityUpdateFrame priority_update; |
| quic::HttpStreamPriority priority{ |
| spdy_priority, quic::HttpStreamPriority::kDefaultIncremental}; |
| if (client_priority_uses_incremental_) { |
| priority.incremental = kDefaultPriorityIncremental; |
| } |
| |
| if (priority.urgency != quic::HttpStreamPriority::kDefaultUrgency || |
| priority.incremental != quic::HttpStreamPriority::kDefaultIncremental) { |
| priority_update.priority_field_value = |
| quic::SerializePriorityFieldValue(priority); |
| } |
| |
| // Only generate a frame if a non-empty string was generated. |
| if (!priority_update.priority_field_value.empty()) { |
| priority_update.prioritized_element_id = stream_id; |
| priority_data = |
| quic::HttpEncoder::SerializePriorityUpdateFrame(priority_update); |
| } |
| |
| return priority_data; |
| } |
| |
| void QuicTestPacketMaker::AddPriorityHeader(spdy::SpdyPriority spdy_priority, |
| spdy::Http2HeaderBlock* headers) { |
| if (use_priority_header_ && |
| base::FeatureList::IsEnabled(net::features::kPriorityHeader)) { |
| quic::HttpStreamPriority priority{ |
| spdy_priority, quic::HttpStreamPriority::kDefaultIncremental}; |
| if (client_priority_uses_incremental_) { |
| priority.incremental = kDefaultPriorityIncremental; |
| } |
| (*headers)[net::kHttp2PriorityHeader] = |
| quic::SerializePriorityFieldValue(priority); |
| } |
| } |
| |
| std::string QuicTestPacketMaker::GenerateHttp3GreaseData() { |
| return quic::HttpEncoder::SerializeGreasingFrame(); |
| } |
| |
| void QuicTestPacketMaker::MaybeAddHttp3SettingsFrames() { |
| quic::QuicStreamId stream_id = |
| quic::QuicUtils::GetFirstUnidirectionalStreamId( |
| version_.transport_version, perspective_); |
| |
| if (stream_offsets_[stream_id] != 0) |
| return; |
| |
| // A stream frame containing stream type will be written on the control |
| // stream first. |
| std::string type(1, 0x00); |
| std::string settings_data = GenerateHttp3SettingsData(); |
| std::string grease_data = GenerateHttp3GreaseData(); |
| |
| // The type and the SETTINGS frame may be sent in multiple QUIC STREAM |
| // frames. |
| std::string data = type + settings_data + grease_data; |
| |
| AddQuicStreamFrame(stream_id, false, data); |
| } |
| |
| bool QuicTestPacketMaker::MaybeCoalesceStreamFrame( |
| const quic::QuicFrame& frame) { |
| if (frames_.empty()) { |
| return false; |
| } |
| if (frame.type != quic::STREAM_FRAME || |
| frames_.back().type != quic::STREAM_FRAME) { |
| return false; |
| } |
| |
| // Make sure they are congruent data segments in the stream. |
| const quic::QuicStreamFrame* new_frame = &frame.stream_frame; |
| quic::QuicStreamFrame* previous_frame = &frames_.back().stream_frame; |
| if (new_frame->stream_id != previous_frame->stream_id || |
| new_frame->offset != |
| previous_frame->offset + previous_frame->data_length) { |
| return false; |
| } |
| |
| // Extend the data buffer to include the data from both frames (into a copy |
| // buffer). This doesn't attempt to limit coalescing to a particular packet |
| // size limit and may need to be updated if a test comes along that |
| // retransmits enough stream data to span multiple packets. |
| std::string data(previous_frame->data_buffer, previous_frame->data_length); |
| data += std::string(new_frame->data_buffer, new_frame->data_length); |
| saved_stream_data_.push_back(std::make_unique<std::string>(data)); |
| std::string_view saved_data = *saved_stream_data_.back(); |
| previous_frame->data_buffer = saved_data.data(); |
| previous_frame->data_length = saved_data.length(); |
| |
| // Copy the fin state from the last frame. |
| previous_frame->fin = new_frame->fin; |
| |
| return true; |
| } |
| |
| } // namespace net::test |