// Copyright 2020 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 "cast/streaming/receiver_message.h" #include #include "absl/strings/ascii.h" #include "absl/types/optional.h" #include "cast/streaming/message_fields.h" #include "json/reader.h" #include "json/writer.h" #include "platform/base/error.h" #include "util/base64.h" #include "util/enum_name_table.h" #include "util/json/json_helpers.h" #include "util/json/json_serialization.h" namespace openscreen { namespace cast { namespace { EnumNameTable kMessageTypeNames{ {{kMessageTypeAnswer, ReceiverMessage::Type::kAnswer}, {"CAPABILITIES_RESPONSE", ReceiverMessage::Type::kCapabilitiesResponse}, {"RPC", ReceiverMessage::Type::kRpc}}}; EnumNameTable kMediaCapabilityNames{{ {"audio", MediaCapability::kAudio}, {"aac", MediaCapability::kAac}, {"opus", MediaCapability::kOpus}, {"video", MediaCapability::kVideo}, {"4k", MediaCapability::k4k}, {"h264", MediaCapability::kH264}, {"vp8", MediaCapability::kVp8}, {"vp9", MediaCapability::kVp9}, {"hevc", MediaCapability::kHevc}, }}; ReceiverMessage::Type GetMessageType(const Json::Value& root) { std::string type; if (!json::ParseAndValidateString(root[kMessageType], &type)) { return ReceiverMessage::Type::kUnknown; } absl::AsciiStrToUpper(&type); ErrorOr parsed = GetEnum(kMessageTypeNames, type); return parsed.value(ReceiverMessage::Type::kUnknown); } bool ParseAndValidateCapability(const Json::Value& value, MediaCapability* out) { std::string c; if (!json::ParseAndValidateString(value, &c)) { return false; } const ErrorOr capability = GetEnum(kMediaCapabilityNames, c); if (capability.is_error()) { return false; } *out = capability.value(); return true; } } // namespace // static ErrorOr ReceiverError::Parse(const Json::Value& value) { if (!value) { return Error(Error::Code::kParameterInvalid, "Empty JSON in receiver error parsing"); } int code; std::string description; if (!json::ParseAndValidateInt(value[kErrorCode], &code) || !json::ParseAndValidateString(value[kErrorDescription], &description)) { return Error::Code::kJsonParseError; } return ReceiverError{code, description}; } Json::Value ReceiverError::ToJson() const { Json::Value root; root[kErrorCode] = code; root[kErrorDescription] = description; return root; } // static ErrorOr ReceiverCapability::Parse( const Json::Value& value) { if (!value) { return Error(Error::Code::kParameterInvalid, "Empty JSON in capabilities parsing"); } int remoting_version; if (!json::ParseAndValidateInt(value["remoting"], &remoting_version)) { remoting_version = ReceiverCapability::kRemotingVersionUnknown; } std::vector capabilities; if (!json::ParseAndValidateArray( value["mediaCaps"], ParseAndValidateCapability, &capabilities)) { return Error(Error::Code::kJsonParseError, "Failed to parse media capabilities"); } return ReceiverCapability{remoting_version, std::move(capabilities)}; } Json::Value ReceiverCapability::ToJson() const { Json::Value root; root["remoting"] = remoting_version; Json::Value capabilities(Json::ValueType::arrayValue); for (const auto& capability : media_capabilities) { capabilities.append(GetEnumName(kMediaCapabilityNames, capability).value()); } root["mediaCaps"] = std::move(capabilities); return root; } // static ErrorOr ReceiverMessage::Parse(const Json::Value& value) { ReceiverMessage message; if (!value || !json::ParseAndValidateInt(value[kSequenceNumber], &(message.sequence_number))) { return Error(Error::Code::kJsonParseError, "Failed to parse sequence number"); } std::string result; if (!json::ParseAndValidateString(value[kResult], &result)) { result = kResultError; } message.type = GetMessageType(value); message.valid = (result == kResultOk || message.type == ReceiverMessage::Type::kRpc); if (!message.valid) { ErrorOr error = ReceiverError::Parse(value[kErrorMessageBody]); if (error.is_value()) { message.body = std::move(error.value()); } return message; } switch (message.type) { case Type::kAnswer: { Answer answer; if (openscreen::cast::Answer::ParseAndValidate(value[kAnswerMessageBody], &answer)) { message.body = std::move(answer); message.valid = true; } } break; case Type::kCapabilitiesResponse: { ErrorOr capability = ReceiverCapability::Parse(value[kCapabilitiesMessageBody]); if (capability.is_value()) { message.body = std::move(capability.value()); message.valid = true; } } break; case Type::kRpc: { std::string encoded_rpc; std::vector rpc; if (json::ParseAndValidateString(value[kRpcMessageBody], &encoded_rpc) && base64::Decode(encoded_rpc, &rpc)) { message.body = std::move(rpc); message.valid = true; } } break; default: break; } return message; } ErrorOr ReceiverMessage::ToJson() const { OSP_CHECK(type != ReceiverMessage::Type::kUnknown) << "Trying to send an unknown message is a developer error"; Json::Value root; root[kMessageType] = GetEnumName(kMessageTypeNames, type).value(); if (sequence_number >= 0) { root[kSequenceNumber] = sequence_number; } switch (type) { case ReceiverMessage::Type::kAnswer: if (valid) { root[kResult] = kResultOk; root[kAnswerMessageBody] = absl::get(body).ToJson(); } else { root[kResult] = kResultError; root[kErrorMessageBody] = absl::get(body).ToJson(); } break; case ReceiverMessage::Type::kCapabilitiesResponse: if (valid) { root[kResult] = kResultOk; root[kCapabilitiesMessageBody] = absl::get(body).ToJson(); } else { root[kResult] = kResultError; root[kErrorMessageBody] = absl::get(body).ToJson(); } break; // NOTE: RPC messages do NOT have a result field. case ReceiverMessage::Type::kRpc: root[kRpcMessageBody] = base64::Encode(absl::get>(body)); break; default: OSP_NOTREACHED(); } return root; } } // namespace cast } // namespace openscreen