| // Copyright (c) 2021 The Chromium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #ifndef QUICHE_COMMON_CAPSULE_H_ |
| #define QUICHE_COMMON_CAPSULE_H_ |
| |
| #include <cstdint> |
| #include <string> |
| #include <vector> |
| |
| #include "absl/strings/string_view.h" |
| #include "absl/types/optional.h" |
| #include "quiche/common/platform/api/quiche_logging.h" |
| #include "quiche/common/quiche_buffer_allocator.h" |
| #include "quiche/common/quiche_ip_address.h" |
| #include "quiche/web_transport/web_transport.h" |
| |
| namespace quiche { |
| |
| enum class CapsuleType : uint64_t { |
| // Casing in this enum matches the IETF specifications. |
| DATAGRAM = 0x00, // RFC 9297. |
| LEGACY_DATAGRAM = 0xff37a0, // draft-ietf-masque-h3-datagram-04. |
| LEGACY_DATAGRAM_WITHOUT_CONTEXT = |
| 0xff37a5, // draft-ietf-masque-h3-datagram-05 to -08. |
| CLOSE_WEBTRANSPORT_SESSION = 0x2843, |
| // draft-ietf-masque-connect-ip-03. |
| ADDRESS_ASSIGN = 0x1ECA6A00, |
| ADDRESS_REQUEST = 0x1ECA6A01, |
| ROUTE_ADVERTISEMENT = 0x1ECA6A02, |
| }; |
| |
| QUICHE_EXPORT std::string CapsuleTypeToString(CapsuleType capsule_type); |
| QUICHE_EXPORT std::ostream& operator<<(std::ostream& os, |
| const CapsuleType& capsule_type); |
| |
| struct QUICHE_EXPORT DatagramCapsule { |
| absl::string_view http_datagram_payload; |
| std::string ToString() const; |
| }; |
| struct QUICHE_EXPORT LegacyDatagramCapsule { |
| absl::string_view http_datagram_payload; |
| std::string ToString() const; |
| }; |
| struct QUICHE_EXPORT LegacyDatagramWithoutContextCapsule { |
| absl::string_view http_datagram_payload; |
| std::string ToString() const; |
| }; |
| struct QUICHE_EXPORT CloseWebTransportSessionCapsule { |
| webtransport::SessionErrorCode error_code; |
| absl::string_view error_message; |
| std::string ToString() const; |
| }; |
| struct QUICHE_EXPORT PrefixWithId { |
| uint64_t request_id; |
| quiche::QuicheIpPrefix ip_prefix; |
| bool operator==(const PrefixWithId& other) const; |
| }; |
| struct QUICHE_EXPORT IpAddressRange { |
| quiche::QuicheIpAddress start_ip_address; |
| quiche::QuicheIpAddress end_ip_address; |
| uint8_t ip_protocol; |
| bool operator==(const IpAddressRange& other) const; |
| }; |
| struct QUICHE_EXPORT AddressAssignCapsule { |
| std::vector<PrefixWithId> assigned_addresses; |
| bool operator==(const AddressAssignCapsule& other) const; |
| std::string ToString() const; |
| }; |
| struct QUICHE_EXPORT AddressRequestCapsule { |
| std::vector<PrefixWithId> requested_addresses; |
| bool operator==(const AddressRequestCapsule& other) const; |
| std::string ToString() const; |
| }; |
| struct QUICHE_EXPORT RouteAdvertisementCapsule { |
| std::vector<IpAddressRange> ip_address_ranges; |
| bool operator==(const RouteAdvertisementCapsule& other) const; |
| std::string ToString() const; |
| }; |
| |
| // Capsule from RFC 9297. |
| // IMPORTANT NOTE: Capsule does not own any of the absl::string_view memory it |
| // points to. Strings saved into a capsule must outlive the capsule object. Any |
| // code that sees a capsule in a callback needs to either process it immediately |
| // or perform its own deep copy. |
| class QUICHE_EXPORT Capsule { |
| public: |
| static Capsule Datagram( |
| absl::string_view http_datagram_payload = absl::string_view()); |
| static Capsule LegacyDatagram( |
| absl::string_view http_datagram_payload = absl::string_view()); |
| static Capsule LegacyDatagramWithoutContext( |
| absl::string_view http_datagram_payload = absl::string_view()); |
| static Capsule CloseWebTransportSession( |
| webtransport::SessionErrorCode error_code = 0, |
| absl::string_view error_message = ""); |
| static Capsule AddressRequest(); |
| static Capsule AddressAssign(); |
| static Capsule RouteAdvertisement(); |
| static Capsule Unknown( |
| uint64_t capsule_type, |
| absl::string_view unknown_capsule_data = absl::string_view()); |
| |
| explicit Capsule(CapsuleType capsule_type); |
| ~Capsule(); |
| Capsule(const Capsule& other); |
| Capsule& operator=(const Capsule& other); |
| bool operator==(const Capsule& other) const; |
| |
| // Human-readable information string for debugging purposes. |
| std::string ToString() const; |
| friend QUICHE_EXPORT std::ostream& operator<<(std::ostream& os, |
| const Capsule& capsule); |
| |
| CapsuleType capsule_type() const { return capsule_type_; } |
| DatagramCapsule& datagram_capsule() { |
| QUICHE_DCHECK_EQ(capsule_type_, CapsuleType::DATAGRAM); |
| return datagram_capsule_; |
| } |
| const DatagramCapsule& datagram_capsule() const { |
| QUICHE_DCHECK_EQ(capsule_type_, CapsuleType::DATAGRAM); |
| return datagram_capsule_; |
| } |
| LegacyDatagramCapsule& legacy_datagram_capsule() { |
| QUICHE_DCHECK_EQ(capsule_type_, CapsuleType::LEGACY_DATAGRAM); |
| return legacy_datagram_capsule_; |
| } |
| const LegacyDatagramCapsule& legacy_datagram_capsule() const { |
| QUICHE_DCHECK_EQ(capsule_type_, CapsuleType::LEGACY_DATAGRAM); |
| return legacy_datagram_capsule_; |
| } |
| LegacyDatagramWithoutContextCapsule& |
| legacy_datagram_without_context_capsule() { |
| QUICHE_DCHECK_EQ(capsule_type_, |
| CapsuleType::LEGACY_DATAGRAM_WITHOUT_CONTEXT); |
| return legacy_datagram_without_context_capsule_; |
| } |
| const LegacyDatagramWithoutContextCapsule& |
| legacy_datagram_without_context_capsule() const { |
| QUICHE_DCHECK_EQ(capsule_type_, |
| CapsuleType::LEGACY_DATAGRAM_WITHOUT_CONTEXT); |
| return legacy_datagram_without_context_capsule_; |
| } |
| CloseWebTransportSessionCapsule& close_web_transport_session_capsule() { |
| QUICHE_DCHECK_EQ(capsule_type_, CapsuleType::CLOSE_WEBTRANSPORT_SESSION); |
| return close_web_transport_session_capsule_; |
| } |
| const CloseWebTransportSessionCapsule& close_web_transport_session_capsule() |
| const { |
| QUICHE_DCHECK_EQ(capsule_type_, CapsuleType::CLOSE_WEBTRANSPORT_SESSION); |
| return close_web_transport_session_capsule_; |
| } |
| AddressRequestCapsule& address_request_capsule() { |
| QUICHE_DCHECK_EQ(capsule_type_, CapsuleType::ADDRESS_REQUEST); |
| return *address_request_capsule_; |
| } |
| const AddressRequestCapsule& address_request_capsule() const { |
| QUICHE_DCHECK_EQ(capsule_type_, CapsuleType::ADDRESS_REQUEST); |
| return *address_request_capsule_; |
| } |
| AddressAssignCapsule& address_assign_capsule() { |
| QUICHE_DCHECK_EQ(capsule_type_, CapsuleType::ADDRESS_ASSIGN); |
| return *address_assign_capsule_; |
| } |
| const AddressAssignCapsule& address_assign_capsule() const { |
| QUICHE_DCHECK_EQ(capsule_type_, CapsuleType::ADDRESS_ASSIGN); |
| return *address_assign_capsule_; |
| } |
| RouteAdvertisementCapsule& route_advertisement_capsule() { |
| QUICHE_DCHECK_EQ(capsule_type_, CapsuleType::ROUTE_ADVERTISEMENT); |
| return *route_advertisement_capsule_; |
| } |
| const RouteAdvertisementCapsule& route_advertisement_capsule() const { |
| QUICHE_DCHECK_EQ(capsule_type_, CapsuleType::ROUTE_ADVERTISEMENT); |
| return *route_advertisement_capsule_; |
| } |
| absl::string_view& unknown_capsule_data() { |
| QUICHE_DCHECK(capsule_type_ != CapsuleType::DATAGRAM && |
| capsule_type_ != CapsuleType::LEGACY_DATAGRAM && |
| capsule_type_ != |
| CapsuleType::LEGACY_DATAGRAM_WITHOUT_CONTEXT && |
| capsule_type_ != CapsuleType::CLOSE_WEBTRANSPORT_SESSION && |
| capsule_type_ != CapsuleType::ADDRESS_REQUEST && |
| capsule_type_ != CapsuleType::ADDRESS_ASSIGN && |
| capsule_type_ != CapsuleType::ROUTE_ADVERTISEMENT) |
| << capsule_type_; |
| return unknown_capsule_data_; |
| } |
| const absl::string_view& unknown_capsule_data() const { |
| QUICHE_DCHECK(capsule_type_ != CapsuleType::DATAGRAM && |
| capsule_type_ != CapsuleType::LEGACY_DATAGRAM && |
| capsule_type_ != |
| CapsuleType::LEGACY_DATAGRAM_WITHOUT_CONTEXT && |
| capsule_type_ != CapsuleType::CLOSE_WEBTRANSPORT_SESSION && |
| capsule_type_ != CapsuleType::ADDRESS_REQUEST && |
| capsule_type_ != CapsuleType::ADDRESS_ASSIGN && |
| capsule_type_ != CapsuleType::ROUTE_ADVERTISEMENT) |
| << capsule_type_; |
| return unknown_capsule_data_; |
| } |
| |
| private: |
| void Free(); |
| CapsuleType capsule_type_; |
| union { |
| DatagramCapsule datagram_capsule_; |
| LegacyDatagramCapsule legacy_datagram_capsule_; |
| LegacyDatagramWithoutContextCapsule |
| legacy_datagram_without_context_capsule_; |
| CloseWebTransportSessionCapsule close_web_transport_session_capsule_; |
| AddressRequestCapsule* address_request_capsule_; |
| AddressAssignCapsule* address_assign_capsule_; |
| RouteAdvertisementCapsule* route_advertisement_capsule_; |
| absl::string_view unknown_capsule_data_; |
| }; |
| }; |
| |
| namespace test { |
| class CapsuleParserPeer; |
| } // namespace test |
| |
| class QUICHE_EXPORT CapsuleParser { |
| public: |
| class QUICHE_EXPORT Visitor { |
| public: |
| virtual ~Visitor() {} |
| |
| // Called when a capsule has been successfully parsed. The return value |
| // indicates whether the contents of the capsule are valid: if false is |
| // returned, the parse operation will be considered failed and |
| // OnCapsuleParseFailure will be called. Note that since Capsule does not |
| // own the memory backing its string_views, that memory is only valid until |
| // this callback returns. Visitors that wish to access the capsule later |
| // MUST make a deep copy before this returns. |
| virtual bool OnCapsule(const Capsule& capsule) = 0; |
| |
| virtual void OnCapsuleParseFailure(const std::string& error_message) = 0; |
| }; |
| |
| // |visitor| must be non-null, and must outlive CapsuleParser. |
| explicit CapsuleParser(Visitor* visitor); |
| |
| // Ingests a capsule fragment (any fragment of bytes from the capsule data |
| // stream) and parses and complete capsules it encounters. Returns false if a |
| // parsing error occurred. |
| bool IngestCapsuleFragment(absl::string_view capsule_fragment); |
| |
| void ErrorIfThereIsRemainingBufferedData(); |
| |
| friend class test::CapsuleParserPeer; |
| |
| private: |
| // Attempts to parse a single capsule from |buffered_data_|. If a full capsule |
| // is not available, returns 0. If a parsing error occurs, returns 0. |
| // Otherwise, returns the number of bytes in the parsed capsule. |
| size_t AttemptParseCapsule(); |
| void ReportParseFailure(const std::string& error_message); |
| |
| // Whether a parsing error has occurred. |
| bool parsing_error_occurred_ = false; |
| // Visitor which will receive callbacks, unowned. |
| Visitor* visitor_; |
| |
| std::string buffered_data_; |
| }; |
| |
| // Serializes |capsule| into a newly allocated buffer. |
| QUICHE_EXPORT quiche::QuicheBuffer SerializeCapsule( |
| const Capsule& capsule, quiche::QuicheBufferAllocator* allocator); |
| |
| } // namespace quiche |
| |
| #endif // QUICHE_COMMON_CAPSULE_H_ |