|
|
|
// Protocol Buffers - Google's data interchange format
|
|
|
|
// Copyright 2023 Google LLC. All rights reserved.
|
|
|
|
//
|
|
|
|
// Use of this source code is governed by a BSD-style
|
|
|
|
// license that can be found in the LICENSE file or at
|
|
|
|
// https://developers.google.com/open-source/licenses/bsd
|
|
|
|
|
|
|
|
#ifndef PROTOBUF_HPB_HPB_H_
|
|
|
|
#define PROTOBUF_HPB_HPB_H_
|
|
|
|
|
|
|
|
#include <cstdint>
|
|
|
|
#include <type_traits>
|
|
|
|
|
|
|
|
#include "absl/status/status.h"
|
|
|
|
#include "absl/status/statusor.h"
|
|
|
|
#include "absl/strings/string_view.h"
|
|
|
|
#include "google/protobuf/hpb/arena.h"
|
|
|
|
#include "google/protobuf/hpb/backend/upb/interop.h"
|
|
|
|
#include "google/protobuf/hpb/extension.h"
|
|
|
|
#include "google/protobuf/hpb/internal/internal.h"
|
|
|
|
#include "google/protobuf/hpb/internal/message_lock.h"
|
|
|
|
#include "google/protobuf/hpb/internal/template_help.h"
|
|
|
|
#include "google/protobuf/hpb/ptr.h"
|
|
|
|
#include "google/protobuf/hpb/status.h"
|
|
|
|
#include "upb/mem/arena.hpp"
|
|
|
|
#include "upb/mini_table/extension.h"
|
|
|
|
#include "upb/wire/decode.h"
|
|
|
|
|
|
|
|
#ifdef HPB_BACKEND_UPB
|
|
|
|
#include "google/protobuf/hpb/backend/upb/upb.h"
|
|
|
|
#else
|
|
|
|
#error hpb backend must be specified
|
|
|
|
#endif
|
|
|
|
|
|
|
|
namespace hpb {
|
|
|
|
class ExtensionRegistry;
|
|
|
|
|
|
|
|
namespace internal {
|
|
|
|
|
|
|
|
absl::Status MoveExtension(upb_Message* message, upb_Arena* message_arena,
|
|
|
|
const upb_MiniTableExtension* ext,
|
|
|
|
upb_Message* extension, upb_Arena* extension_arena);
|
|
|
|
|
|
|
|
absl::Status SetExtension(upb_Message* message, upb_Arena* message_arena,
|
|
|
|
const upb_MiniTableExtension* ext,
|
|
|
|
const upb_Message* extension);
|
|
|
|
|
|
|
|
} // namespace internal
|
|
|
|
|
|
|
|
#ifdef HPB_BACKEND_UPB
|
|
|
|
namespace backend = ::hpb::internal::backend::upb;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
template <typename T, typename Extendee, typename Extension,
|
|
|
|
typename = hpb::internal::EnableIfHpbClass<T>>
|
|
|
|
ABSL_MUST_USE_RESULT bool HasExtension(
|
|
|
|
Ptr<T> message,
|
|
|
|
const ::hpb::internal::ExtensionIdentifier<Extendee, Extension>& id) {
|
|
|
|
return ::hpb::internal::HasExtensionOrUnknown(
|
|
|
|
hpb::interop::upb::GetMessage(message), id.mini_table_ext());
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T, typename Extendee, typename Extension,
|
|
|
|
typename = hpb::internal::EnableIfHpbClass<T>>
|
|
|
|
ABSL_MUST_USE_RESULT bool HasExtension(
|
|
|
|
const T* message,
|
|
|
|
const ::hpb::internal::ExtensionIdentifier<Extendee, Extension>& id) {
|
|
|
|
return HasExtension(Ptr(message), id);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T, typename Extension,
|
|
|
|
typename = hpb::internal::EnableIfHpbClass<T>,
|
|
|
|
typename = hpb::internal::EnableIfMutableProto<T>>
|
|
|
|
void ClearExtension(
|
|
|
|
Ptr<T> message,
|
|
|
|
const ::hpb::internal::ExtensionIdentifier<T, Extension>& id) {
|
|
|
|
static_assert(!std::is_const_v<T>, "");
|
|
|
|
upb_Message_ClearExtension(hpb::interop::upb::GetMessage(message),
|
|
|
|
id.mini_table_ext());
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T, typename Extension,
|
|
|
|
typename = hpb::internal::EnableIfHpbClass<T>>
|
|
|
|
void ClearExtension(
|
|
|
|
T* message, const ::hpb::internal::ExtensionIdentifier<T, Extension>& id) {
|
|
|
|
ClearExtension(Ptr(message), id);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T, typename Extension,
|
|
|
|
typename = hpb::internal::EnableIfHpbClass<T>,
|
|
|
|
typename = hpb::internal::EnableIfMutableProto<T>>
|
|
|
|
absl::Status SetExtension(
|
|
|
|
Ptr<T> message,
|
|
|
|
const ::hpb::internal::ExtensionIdentifier<T, Extension>& id,
|
|
|
|
const Extension& value) {
|
|
|
|
static_assert(!std::is_const_v<T>);
|
|
|
|
auto* message_arena = hpb::interop::upb::GetArena(message);
|
|
|
|
return ::hpb::internal::SetExtension(hpb::interop::upb::GetMessage(message),
|
|
|
|
message_arena, id.mini_table_ext(),
|
|
|
|
hpb::interop::upb::GetMessage(&value));
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T, typename Extension,
|
|
|
|
typename = hpb::internal::EnableIfHpbClass<T>,
|
|
|
|
typename = hpb::internal::EnableIfMutableProto<T>>
|
|
|
|
absl::Status SetExtension(
|
|
|
|
Ptr<T> message,
|
|
|
|
const ::hpb::internal::ExtensionIdentifier<T, Extension>& id,
|
|
|
|
Ptr<Extension> value) {
|
|
|
|
static_assert(!std::is_const_v<T>);
|
|
|
|
auto* message_arena = hpb::interop::upb::GetArena(message);
|
|
|
|
return ::hpb::internal::SetExtension(hpb::interop::upb::GetMessage(message),
|
|
|
|
message_arena, id.mini_table_ext(),
|
|
|
|
hpb::interop::upb::GetMessage(value));
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T, typename Extension,
|
|
|
|
typename = hpb::internal::EnableIfHpbClass<T>,
|
|
|
|
typename = hpb::internal::EnableIfMutableProto<T>>
|
|
|
|
absl::Status SetExtension(
|
|
|
|
Ptr<T> message,
|
|
|
|
const ::hpb::internal::ExtensionIdentifier<T, Extension>& id,
|
|
|
|
Extension&& value) {
|
|
|
|
Extension ext = std::move(value);
|
|
|
|
static_assert(!std::is_const_v<T>);
|
|
|
|
auto* message_arena = hpb::interop::upb::GetArena(message);
|
|
|
|
auto* extension_arena = hpb::interop::upb::GetArena(&ext);
|
|
|
|
return ::hpb::internal::MoveExtension(hpb::interop::upb::GetMessage(message),
|
|
|
|
message_arena, id.mini_table_ext(),
|
|
|
|
hpb::interop::upb::GetMessage(&ext),
|
|
|
|
extension_arena);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T, typename Extension,
|
|
|
|
typename = hpb::internal::EnableIfHpbClass<T>>
|
|
|
|
absl::Status SetExtension(
|
|
|
|
T* message, const ::hpb::internal::ExtensionIdentifier<T, Extension>& id,
|
|
|
|
const Extension& value) {
|
|
|
|
return ::hpb::SetExtension(Ptr(message), id, value);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T, typename Extension,
|
|
|
|
typename = hpb::internal::EnableIfHpbClass<T>>
|
|
|
|
absl::Status SetExtension(
|
|
|
|
T* message, const ::hpb::internal::ExtensionIdentifier<T, Extension>& id,
|
|
|
|
Extension&& value) {
|
|
|
|
return ::hpb::SetExtension(Ptr(message), id, std::forward<Extension>(value));
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T, typename Extension,
|
|
|
|
typename = hpb::internal::EnableIfHpbClass<T>>
|
|
|
|
absl::Status SetExtension(
|
|
|
|
T* message, const ::hpb::internal::ExtensionIdentifier<T, Extension>& id,
|
|
|
|
Ptr<Extension> value) {
|
|
|
|
return ::hpb::SetExtension(Ptr(message), id, value);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T, typename Extendee, typename Extension,
|
|
|
|
typename = hpb::internal::EnableIfHpbClass<T>>
|
|
|
|
absl::StatusOr<Ptr<const Extension>> GetExtension(
|
|
|
|
Ptr<T> message,
|
|
|
|
const ::hpb::internal::ExtensionIdentifier<Extendee, Extension>& id) {
|
|
|
|
upb_MessageValue value;
|
|
|
|
const bool ok = ::hpb::internal::GetOrPromoteExtension(
|
|
|
|
hpb::interop::upb::GetMessage(message), id.mini_table_ext(),
|
|
|
|
hpb::interop::upb::GetArena(message), &value);
|
|
|
|
if (!ok) {
|
|
|
|
return ExtensionNotFoundError(
|
|
|
|
upb_MiniTableExtension_Number(id.mini_table_ext()));
|
|
|
|
}
|
|
|
|
return Ptr<const Extension>(::hpb::interop::upb::MakeCHandle<Extension>(
|
|
|
|
(upb_Message*)value.msg_val, hpb::interop::upb::GetArena(message)));
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T, typename Extendee, typename Extension,
|
|
|
|
typename = hpb::internal::EnableIfHpbClass<T>>
|
|
|
|
absl::StatusOr<Ptr<const Extension>> GetExtension(
|
|
|
|
const T* message,
|
|
|
|
const ::hpb::internal::ExtensionIdentifier<Extendee, Extension>& id) {
|
|
|
|
return GetExtension(Ptr(message), id);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T, typename Extension>
|
|
|
|
constexpr uint32_t ExtensionNumber(
|
|
|
|
::hpb::internal::ExtensionIdentifier<T, Extension> id) {
|
|
|
|
return ::hpb::internal::PrivateAccess::GetExtensionNumber(id);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
typename T::Proxy CreateMessage(::hpb::Arena& arena) {
|
|
|
|
return typename T::Proxy(upb_Message_New(T::minitable(), arena.ptr()),
|
|
|
|
arena.ptr());
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
typename T::Proxy CloneMessage(Ptr<T> message, upb_Arena* arena) {
|
|
|
|
return ::hpb::internal::PrivateAccess::Proxy<T>(
|
|
|
|
::hpb::internal::DeepClone(hpb::interop::upb::GetMessage(message),
|
|
|
|
T::minitable(), arena),
|
|
|
|
arena);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
void DeepCopy(Ptr<const T> source_message, Ptr<T> target_message) {
|
|
|
|
static_assert(!std::is_const_v<T>);
|
|
|
|
::hpb::internal::DeepCopy(hpb::interop::upb::GetMessage(target_message),
|
|
|
|
hpb::interop::upb::GetMessage(source_message),
|
|
|
|
T::minitable(),
|
|
|
|
hpb::interop::upb::GetArena(target_message));
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
void DeepCopy(Ptr<const T> source_message, T* target_message) {
|
|
|
|
static_assert(!std::is_const_v<T>);
|
|
|
|
DeepCopy(source_message, Ptr(target_message));
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
void DeepCopy(const T* source_message, Ptr<T> target_message) {
|
|
|
|
static_assert(!std::is_const_v<T>);
|
|
|
|
DeepCopy(Ptr(source_message), target_message);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
void DeepCopy(const T* source_message, T* target_message) {
|
|
|
|
static_assert(!std::is_const_v<T>);
|
|
|
|
DeepCopy(Ptr(source_message), Ptr(target_message));
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
void ClearMessage(hpb::internal::PtrOrRaw<T> message) {
|
|
|
|
backend::ClearMessage(message);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
ABSL_MUST_USE_RESULT bool Parse(Ptr<T> message, absl::string_view bytes) {
|
|
|
|
static_assert(!std::is_const_v<T>);
|
|
|
|
upb_Message_Clear(hpb::interop::upb::GetMessage(message),
|
|
|
|
::hpb::interop::upb::GetMiniTable(message));
|
|
|
|
auto* arena = hpb::interop::upb::GetArena(message);
|
|
|
|
return upb_Decode(bytes.data(), bytes.size(),
|
|
|
|
hpb::interop::upb::GetMessage(message),
|
|
|
|
::hpb::interop::upb::GetMiniTable(message),
|
|
|
|
/* extreg= */ nullptr, /* options= */ 0,
|
|
|
|
arena) == kUpb_DecodeStatus_Ok;
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
ABSL_MUST_USE_RESULT bool Parse(
|
|
|
|
Ptr<T> message, absl::string_view bytes,
|
|
|
|
const ::hpb::ExtensionRegistry& extension_registry) {
|
|
|
|
static_assert(!std::is_const_v<T>);
|
|
|
|
upb_Message_Clear(hpb::interop::upb::GetMessage(message),
|
|
|
|
::hpb::interop::upb::GetMiniTable(message));
|
|
|
|
auto* arena = hpb::interop::upb::GetArena(message);
|
|
|
|
return upb_Decode(bytes.data(), bytes.size(),
|
|
|
|
hpb::interop::upb::GetMessage(message),
|
|
|
|
::hpb::interop::upb::GetMiniTable(message),
|
|
|
|
/* extreg= */
|
|
|
|
::hpb::internal::GetUpbExtensions(extension_registry),
|
|
|
|
/* options= */ 0, arena) == kUpb_DecodeStatus_Ok;
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
ABSL_MUST_USE_RESULT bool Parse(
|
|
|
|
T* message, absl::string_view bytes,
|
|
|
|
const ::hpb::ExtensionRegistry& extension_registry) {
|
|
|
|
static_assert(!std::is_const_v<T>);
|
|
|
|
return Parse(Ptr(message, bytes, extension_registry));
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
ABSL_MUST_USE_RESULT bool Parse(T* message, absl::string_view bytes) {
|
|
|
|
static_assert(!std::is_const_v<T>);
|
|
|
|
upb_Message_Clear(hpb::interop::upb::GetMessage(message),
|
|
|
|
::hpb::interop::upb::GetMiniTable(message));
|
|
|
|
auto* arena = hpb::interop::upb::GetArena(message);
|
|
|
|
return upb_Decode(bytes.data(), bytes.size(),
|
|
|
|
hpb::interop::upb::GetMessage(message),
|
|
|
|
::hpb::interop::upb::GetMiniTable(message),
|
|
|
|
/* extreg= */ nullptr, /* options= */ 0,
|
|
|
|
arena) == kUpb_DecodeStatus_Ok;
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
absl::StatusOr<T> Parse(absl::string_view bytes, int options = 0) {
|
|
|
|
T message;
|
|
|
|
auto* arena = hpb::interop::upb::GetArena(&message);
|
|
|
|
upb_DecodeStatus status =
|
|
|
|
upb_Decode(bytes.data(), bytes.size(), message.msg(),
|
|
|
|
::hpb::interop::upb::GetMiniTable(&message),
|
|
|
|
/* extreg= */ nullptr, /* options= */ 0, arena);
|
|
|
|
if (status == kUpb_DecodeStatus_Ok) {
|
|
|
|
return message;
|
|
|
|
}
|
|
|
|
return MessageDecodeError(status);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
absl::StatusOr<T> Parse(absl::string_view bytes,
|
|
|
|
const ::hpb::ExtensionRegistry& extension_registry,
|
|
|
|
int options = 0) {
|
|
|
|
T message;
|
|
|
|
auto* arena = hpb::interop::upb::GetArena(&message);
|
|
|
|
upb_DecodeStatus status =
|
|
|
|
upb_Decode(bytes.data(), bytes.size(), message.msg(),
|
|
|
|
::hpb::interop::upb::GetMiniTable(&message),
|
|
|
|
::hpb::internal::GetUpbExtensions(extension_registry),
|
|
|
|
/* options= */ 0, arena);
|
|
|
|
if (status == kUpb_DecodeStatus_Ok) {
|
|
|
|
return message;
|
|
|
|
}
|
|
|
|
return MessageDecodeError(status);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
absl::StatusOr<absl::string_view> Serialize(const T* message, upb::Arena& arena,
|
|
|
|
int options = 0) {
|
|
|
|
return ::hpb::internal::Serialize(hpb::interop::upb::GetMessage(message),
|
|
|
|
::hpb::interop::upb::GetMiniTable(message),
|
|
|
|
arena.ptr(), options);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
absl::StatusOr<absl::string_view> Serialize(Ptr<T> message, upb::Arena& arena,
|
|
|
|
int options = 0) {
|
|
|
|
return ::hpb::internal::Serialize(hpb::interop::upb::GetMessage(message),
|
|
|
|
::hpb::interop::upb::GetMiniTable(message),
|
|
|
|
arena.ptr(), options);
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace hpb
|
|
|
|
|
|
|
|
#endif // PROTOBUF_HPB_HPB_H_
|