blob: 7e6f4806bdd1535272ab4c288fee4f9681b95eea [file] [log] [blame]
// Copyright (c) 2023 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.
#include "quiche/quic/moqt/moqt_messages.h"
#include <array>
#include <cstddef>
#include <cstdint>
#include <optional>
#include <string>
#include <utility>
#include <vector>
#include "absl/algorithm/container.h"
#include "absl/container/btree_map.h"
#include "absl/status/status.h"
#include "absl/strings/escaping.h"
#include "absl/strings/str_cat.h"
#include "absl/strings/str_join.h"
#include "absl/strings/string_view.h"
#include "absl/types/span.h"
#include "quiche/quic/core/quic_types.h"
#include "quiche/quic/platform/api/quic_bug_tracker.h"
#include "quiche/common/platform/api/quiche_bug_tracker.h"
#include "quiche/web_transport/web_transport.h"
namespace moqt {
void KeyValuePairList::insert(uint64_t key, absl::string_view value) {
if (key % 2 == 0) {
QUICHE_BUG(key_value_pair_string_is_even) << "Key value pair of wrong type";
return;
}
string_map_.emplace(key, value);
}
void KeyValuePairList::insert(uint64_t key, uint64_t value) {
if (key % 2 == 1) {
QUICHE_BUG(key_value_pair_int_is_odd) << "Key value pair of wrong type";
return;
}
integer_map_.emplace(key, value);
}
size_t KeyValuePairList::count(uint64_t key) const {
if (key % 2 == 0) {
return integer_map_.count(key);
}
return string_map_.count(key);
}
bool KeyValuePairList::contains(uint64_t key) const {
if (key % 2 == 0) {
return integer_map_.contains(key);
}
return string_map_.contains(key);
}
std::vector<uint64_t> KeyValuePairList::GetIntegers(uint64_t key) const {
if (key % 2 == 1) {
QUICHE_BUG(key_value_pair_int_is_odd) << "Key value pair of wrong type";
return {};
}
std::vector<uint64_t> result;
auto [range_start, range_end] = integer_map_.equal_range(key);
for (auto& it = range_start; it != range_end; ++it) {
result.push_back(it->second);
}
return result;
}
std::vector<absl::string_view> KeyValuePairList::GetStrings(
uint64_t key) const {
if (key % 2 == 0) {
QUICHE_BUG(key_value_pair_string_is_even) << "Key value pair of wrong type";
return {};
}
std::vector<absl::string_view> result;
auto [range_start, range_end] = string_map_.equal_range(key);
for (auto& it = range_start; it != range_end; ++it) {
result.push_back(it->second);
}
return result;
}
MoqtObjectStatus IntegerToObjectStatus(uint64_t integer) {
if (integer >=
static_cast<uint64_t>(MoqtObjectStatus::kInvalidObjectStatus)) {
return MoqtObjectStatus::kInvalidObjectStatus;
}
return static_cast<MoqtObjectStatus>(integer);
}
SubscribeErrorCode StatusToSubscribeErrorCode(absl::Status status) {
QUICHE_DCHECK(!status.ok());
switch (status.code()) {
case absl::StatusCode::kPermissionDenied:
case absl::StatusCode::kUnauthenticated:
return SubscribeErrorCode::kUnauthorized;
case absl::StatusCode::kDeadlineExceeded:
return SubscribeErrorCode::kTimeout;
case absl::StatusCode::kUnavailable:
return SubscribeErrorCode::kNotSupported;
case absl::StatusCode::kNotFound:
return SubscribeErrorCode::kDoesNotExist;
case absl::StatusCode::kOutOfRange:
return SubscribeErrorCode::kInvalidRange;
default:
return SubscribeErrorCode::kInternalError;
}
}
MoqtError ValidateSetupParameters(const KeyValuePairList& parameters,
bool webtrans,
quic::Perspective perspective) {
if (parameters.count(SetupParameter::kPath) > 1 ||
parameters.count(SetupParameter::kMaxRequestId) > 1 ||
parameters.count(SetupParameter::kMaxAuthTokenCacheSize) > 1 ||
parameters.count(SetupParameter::kSupportObjectAcks) > 1) {
return MoqtError::kKeyValueFormattingError;
}
if ((webtrans || perspective == quic::Perspective::IS_CLIENT) ==
parameters.contains(SetupParameter::kPath)) {
// Only non-webtrans servers should receive kPath.
return MoqtError::kInvalidPath;
}
if (!parameters.contains(SetupParameter::kSupportObjectAcks)) {
return MoqtError::kNoError;
}
std::vector<uint64_t> support_object_acks =
parameters.GetIntegers(SetupParameter::kSupportObjectAcks);
QUICHE_DCHECK(support_object_acks.size() == 1);
if (support_object_acks.front() > 1) {
return MoqtError::kKeyValueFormattingError;
}
return MoqtError::kNoError;
}
const std::array<MoqtMessageType, 5> kAllowsAuthorization = {
MoqtMessageType::kSubscribe, MoqtMessageType::kTrackStatusRequest,
MoqtMessageType::kFetch, MoqtMessageType::kSubscribeAnnounces,
MoqtMessageType::kAnnounce};
const std::array<MoqtMessageType, 4> kAllowsDeliveryTimeout = {
MoqtMessageType::kSubscribe, MoqtMessageType::kSubscribeOk,
MoqtMessageType::kSubscribeUpdate, MoqtMessageType::kTrackStatus};
const std::array<MoqtMessageType, 3> kAllowsMaxCacheDuration = {
MoqtMessageType::kSubscribeOk, MoqtMessageType::kTrackStatus,
MoqtMessageType::kFetchOk};
bool ValidateVersionSpecificParameters(const KeyValuePairList& parameters,
MoqtMessageType message_type) {
size_t authorization_token =
parameters.count(VersionSpecificParameter::kAuthorizationToken);
size_t delivery_timeout =
parameters.count(VersionSpecificParameter::kDeliveryTimeout);
size_t max_cache_duration =
parameters.count(VersionSpecificParameter::kMaxCacheDuration);
if (delivery_timeout > 1 || max_cache_duration > 1) {
// Disallowed duplicate.
return false;
}
if (authorization_token > 0 &&
!absl::c_linear_search(kAllowsAuthorization, message_type)) {
return false;
}
if (delivery_timeout > 0 &&
!absl::c_linear_search(kAllowsDeliveryTimeout, message_type)) {
return false;
}
if (max_cache_duration > 0 &&
!absl::c_linear_search(kAllowsMaxCacheDuration, message_type)) {
return false;
}
return true;
}
std::string MoqtMessageTypeToString(const MoqtMessageType message_type) {
switch (message_type) {
case MoqtMessageType::kClientSetup:
return "CLIENT_SETUP";
case MoqtMessageType::kServerSetup:
return "SERVER_SETUP";
case MoqtMessageType::kSubscribe:
return "SUBSCRIBE";
case MoqtMessageType::kSubscribeOk:
return "SUBSCRIBE_OK";
case MoqtMessageType::kSubscribeError:
return "SUBSCRIBE_ERROR";
case MoqtMessageType::kUnsubscribe:
return "UNSUBSCRIBE";
case MoqtMessageType::kSubscribeDone:
return "SUBSCRIBE_DONE";
case MoqtMessageType::kSubscribeUpdate:
return "SUBSCRIBE_UPDATE";
case MoqtMessageType::kAnnounceCancel:
return "ANNOUNCE_CANCEL";
case MoqtMessageType::kTrackStatusRequest:
return "TRACK_STATUS_REQUEST";
case MoqtMessageType::kTrackStatus:
return "TRACK_STATUS";
case MoqtMessageType::kAnnounce:
return "ANNOUNCE";
case MoqtMessageType::kAnnounceOk:
return "ANNOUNCE_OK";
case MoqtMessageType::kAnnounceError:
return "ANNOUNCE_ERROR";
case MoqtMessageType::kUnannounce:
return "UNANNOUNCE";
case MoqtMessageType::kGoAway:
return "GOAWAY";
case MoqtMessageType::kSubscribeAnnounces:
return "SUBSCRIBE_NAMESPACE";
case MoqtMessageType::kSubscribeAnnouncesOk:
return "SUBSCRIBE_NAMESPACE_OK";
case MoqtMessageType::kSubscribeAnnouncesError:
return "SUBSCRIBE_NAMESPACE_ERROR";
case MoqtMessageType::kUnsubscribeAnnounces:
return "UNSUBSCRIBE_NAMESPACE";
case MoqtMessageType::kMaxRequestId:
return "MAX_REQUEST_ID";
case MoqtMessageType::kFetch:
return "FETCH";
case MoqtMessageType::kFetchCancel:
return "FETCH_CANCEL";
case MoqtMessageType::kFetchOk:
return "FETCH_OK";
case MoqtMessageType::kFetchError:
return "FETCH_ERROR";
case MoqtMessageType::kRequestsBlocked:
return "REQUESTS_BLOCKED";
case MoqtMessageType::kObjectAck:
return "OBJECT_ACK";
}
return "Unknown message " + std::to_string(static_cast<int>(message_type));
}
std::string MoqtDataStreamTypeToString(MoqtDataStreamType type) {
switch (type) {
case MoqtDataStreamType::kStreamHeaderSubgroup:
return "STREAM_HEADER_SUBGROUP";
case MoqtDataStreamType::kStreamHeaderFetch:
return "STREAM_HEADER_FETCH";
case MoqtDataStreamType::kPadding:
return "PADDING";
}
return "Unknown stream type " + absl::StrCat(static_cast<int>(type));
}
std::string MoqtDatagramTypeToString(MoqtDatagramType type) {
switch (type) {
case MoqtDatagramType::kObject:
return "OBJECT_DATAGRAM";
case MoqtDatagramType::kObjectStatus:
return "OBJECT_STATUS_DATAGRAM";
}
return "Unknown datagram type " + absl::StrCat(static_cast<int>(type));
}
std::string MoqtForwardingPreferenceToString(
MoqtForwardingPreference preference) {
switch (preference) {
case MoqtForwardingPreference::kDatagram:
return "DATAGRAM";
case MoqtForwardingPreference::kSubgroup:
return "SUBGROUP";
}
QUIC_BUG(quic_bug_bad_moqt_message_type_01)
<< "Unknown preference " << std::to_string(static_cast<int>(preference));
return "Unknown preference " + std::to_string(static_cast<int>(preference));
}
std::string FullTrackName::ToString() const {
std::vector<std::string> bits;
bits.reserve(tuple_.size());
for (absl::string_view raw_bit : tuple_) {
bits.push_back(absl::StrCat("\"", absl::CHexEscape(raw_bit), "\""));
}
return absl::StrCat("{", absl::StrJoin(bits, ", "), "}");
}
bool FullTrackName::operator==(const FullTrackName& other) const {
if (tuple_.size() != other.tuple_.size()) {
return false;
}
return absl::c_equal(tuple_, other.tuple_);
}
bool FullTrackName::operator<(const FullTrackName& other) const {
return absl::c_lexicographical_compare(tuple_, other.tuple_);
}
FullTrackName::FullTrackName(absl::Span<const absl::string_view> elements)
: tuple_(elements.begin(), elements.end()) {
QUICHE_BUG_IF(Moqt_namespace_too_large_03,
std::size(elements) > (kMaxNamespaceElements + 1))
<< "Constructing a namespace that is too large.";
}
absl::Status MoqtStreamErrorToStatus(webtransport::StreamErrorCode error_code,
absl::string_view reason_phrase) {
switch (error_code) {
case kResetCodeSubscriptionGone:
return absl::NotFoundError(reason_phrase);
case kResetCodeTimedOut:
return absl::DeadlineExceededError(reason_phrase);
default:
return absl::UnknownError(reason_phrase);
}
}
} // namespace moqt