Protocol Buffers - Google's data interchange format (grpc依赖)
https://developers.google.com/protocol-buffers/
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
442 lines
15 KiB
442 lines
15 KiB
/* |
|
* Copyright (c) 2009-2021, Google LLC |
|
* All rights reserved. |
|
* |
|
* Redistribution and use in source and binary forms, with or without |
|
* modification, are permitted provided that the following conditions are met: |
|
* * Redistributions of source code must retain the above copyright |
|
* notice, this list of conditions and the following disclaimer. |
|
* * Redistributions in binary form must reproduce the above copyright |
|
* notice, this list of conditions and the following disclaimer in the |
|
* documentation and/or other materials provided with the distribution. |
|
* * Neither the name of Google LLC nor the |
|
* names of its contributors may be used to endorse or promote products |
|
* derived from this software without specific prior written permission. |
|
* |
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" |
|
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE |
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE |
|
* ARE DISCLAIMED. IN NO EVENT SHALL Google LLC BE LIABLE FOR ANY DIRECT, |
|
* INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES |
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; |
|
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND |
|
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS |
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
|
*/ |
|
|
|
#ifndef UPB_PROTOS_PROTOS_H_ |
|
#define UPB_PROTOS_PROTOS_H_ |
|
|
|
#include <type_traits> |
|
|
|
#include "absl/status/status.h" |
|
#include "absl/status/statusor.h" |
|
#include "upb/message/extension_internal.h" |
|
#include "upb/mini_table/types.h" |
|
#include "upb/upb.hpp" |
|
#include "upb/wire/decode.h" |
|
#include "upb/wire/encode.h" |
|
|
|
namespace protos { |
|
|
|
using Arena = ::upb::Arena; |
|
class ExtensionRegistry; |
|
|
|
template <typename T> |
|
using Proxy = std::conditional_t<std::is_const<T>::value, |
|
typename std::remove_const_t<T>::CProxy, |
|
typename T::Proxy>; |
|
|
|
// Provides convenient access to Proxy and CProxy message types. |
|
// |
|
// Using rebinding and handling of const, Ptr<Message> and Ptr<const Message> |
|
// allows copying const with T* const and avoids using non-copyable Proxy types |
|
// directly. |
|
template <typename T> |
|
class Ptr final { |
|
public: |
|
Ptr() = delete; |
|
|
|
// Implicit conversions |
|
Ptr(T* m) : p_(m) {} // NOLINT |
|
Ptr(const Proxy<T>* p) : p_(*p) {} // NOLINT |
|
Ptr(Proxy<T> p) : p_(p) {} // NOLINT |
|
Ptr(const Ptr& m) = default; |
|
|
|
Ptr& operator=(Ptr v) & { |
|
Proxy<T>::Rebind(p_, v.p_); |
|
return *this; |
|
} |
|
|
|
Proxy<T> operator*() const { return p_; } |
|
Proxy<T>* operator->() const { |
|
return const_cast<Proxy<T>*>(std::addressof(p_)); |
|
} |
|
|
|
#ifdef __clang__ |
|
#pragma clang diagnostic push |
|
#pragma clang diagnostic ignored "-Wclass-conversion" |
|
#endif |
|
template <typename U = T, std::enable_if_t<!std::is_const<U>::value, int> = 0> |
|
operator Ptr<const T>() const { |
|
Proxy<const T> p(p_); |
|
return Ptr<const T>(&p); |
|
} |
|
#ifdef __clang__ |
|
#pragma clang diagnostic pop |
|
#endif |
|
|
|
private: |
|
Ptr(void* msg, upb_Arena* arena) : p_(msg, arena) {} // NOLINT |
|
|
|
friend class Ptr<const T>; |
|
friend typename T::Access; |
|
|
|
Proxy<T> p_; |
|
}; |
|
|
|
inline absl::string_view UpbStrToStringView(upb_StringView str) { |
|
return absl::string_view(str.data, str.size); |
|
} |
|
|
|
// TODO: update bzl and move to upb runtime / protos.cc. |
|
inline upb_StringView UpbStrFromStringView(absl::string_view str, |
|
upb_Arena* arena) { |
|
const size_t str_size = str.size(); |
|
char* buffer = static_cast<char*>(upb_Arena_Malloc(arena, str_size)); |
|
memcpy(buffer, str.data(), str_size); |
|
return upb_StringView_FromDataAndSize(buffer, str_size); |
|
} |
|
|
|
template <typename T> |
|
typename T::Proxy CreateMessage(::protos::Arena& arena) { |
|
return typename T::Proxy(upb_Message_New(T::minitable(), arena.ptr()), |
|
arena.ptr()); |
|
} |
|
|
|
// begin:github_only |
|
// This type exists to work around an absl type that has not yet been |
|
// released. |
|
struct SourceLocation { |
|
static SourceLocation current() { return {}; } |
|
absl::string_view file_name() { return "<unknown>"; } |
|
int line() { return 0; } |
|
}; |
|
// end:github_only |
|
|
|
// begin:google_only |
|
// using SourceLocation = absl::SourceLocation; |
|
// end:google_only |
|
|
|
absl::Status MessageAllocationError( |
|
SourceLocation loc = SourceLocation::current()); |
|
|
|
absl::Status ExtensionNotFoundError( |
|
int extension_number, SourceLocation loc = SourceLocation::current()); |
|
|
|
absl::Status MessageDecodeError(upb_DecodeStatus status, |
|
SourceLocation loc = SourceLocation::current()); |
|
|
|
absl::Status MessageEncodeError(upb_EncodeStatus status, |
|
SourceLocation loc = SourceLocation::current()); |
|
|
|
namespace internal { |
|
template <typename T> |
|
T CreateMessage() { |
|
return T(); |
|
} |
|
|
|
template <typename T> |
|
typename T::Proxy CreateMessageProxy(void* msg, upb_Arena* arena) { |
|
return typename T::Proxy(msg, arena); |
|
} |
|
|
|
template <typename T> |
|
typename T::CProxy CreateMessage(upb_Message* msg) { |
|
return typename T::CProxy(msg); |
|
} |
|
|
|
class ExtensionMiniTableProvider { |
|
public: |
|
ExtensionMiniTableProvider(const upb_MiniTableExtension* mini_table_ext) |
|
: mini_table_ext_(mini_table_ext) {} |
|
const upb_MiniTableExtension* mini_table_ext() const { |
|
return mini_table_ext_; |
|
} |
|
|
|
private: |
|
const upb_MiniTableExtension* mini_table_ext_; |
|
}; |
|
|
|
// ------------------------------------------------------------------- |
|
// ExtensionIdentifier |
|
// This is the type of actual extension objects. E.g. if you have: |
|
// extend Foo { |
|
// optional MyExtension bar = 1234; |
|
// } |
|
// then "bar" will be defined in C++ as: |
|
// ExtensionIdentifier<Foo, MyExtension> bar(&namespace_bar_ext); |
|
template <typename ExtendeeType, typename ExtensionType> |
|
class ExtensionIdentifier : public ExtensionMiniTableProvider { |
|
public: |
|
using Extension = ExtensionType; |
|
using Extendee = ExtendeeType; |
|
|
|
ExtensionIdentifier(const upb_MiniTableExtension* mini_table_ext) |
|
: ExtensionMiniTableProvider(mini_table_ext) {} |
|
}; |
|
|
|
template <typename T> |
|
void* GetInternalMsg(const T& message) { |
|
return message.msg(); |
|
} |
|
|
|
template <typename T> |
|
void* GetInternalMsg(const Ptr<T>& message) { |
|
return message->msg(); |
|
} |
|
|
|
template <typename T> |
|
upb_Arena* GetArena(const T& message) { |
|
return static_cast<upb_Arena*>(message.GetInternalArena()); |
|
} |
|
|
|
template <typename T> |
|
upb_Arena* GetArena(const Ptr<T>& message) { |
|
return static_cast<upb_Arena*>(message->GetInternalArena()); |
|
} |
|
|
|
upb_ExtensionRegistry* GetUpbExtensions( |
|
const ExtensionRegistry& extension_registry); |
|
|
|
absl::StatusOr<absl::string_view> Serialize(const upb_Message* message, |
|
const upb_MiniTable* mini_table, |
|
upb_Arena* arena, int options); |
|
|
|
} // namespace internal |
|
|
|
class ExtensionRegistry { |
|
public: |
|
ExtensionRegistry( |
|
const std::vector<const ::protos::internal::ExtensionMiniTableProvider*> |
|
extensions, |
|
const upb::Arena& arena) |
|
: registry_(upb_ExtensionRegistry_New(arena.ptr())) { |
|
if (registry_) { |
|
for (const auto& ext_provider : extensions) { |
|
const auto* ext = ext_provider->mini_table_ext(); |
|
bool success = upb_ExtensionRegistry_AddArray(registry_, &ext, 1); |
|
if (!success) { |
|
registry_ = nullptr; |
|
break; |
|
} |
|
} |
|
} |
|
} |
|
|
|
private: |
|
friend upb_ExtensionRegistry* ::protos::internal::GetUpbExtensions( |
|
const ExtensionRegistry& extension_registry); |
|
upb_ExtensionRegistry* registry_; |
|
}; |
|
|
|
template <typename T> |
|
using EnableIfProtosClass = std::enable_if_t< |
|
std::is_base_of<typename T::Access, T>::value && |
|
std::is_base_of<typename T::Access, typename T::ExtendableType>::value>; |
|
|
|
template <typename T> |
|
using EnableIfMutableProto = std::enable_if_t<!std::is_const<T>::value>; |
|
|
|
template <typename T, typename Extendee, typename Extension, |
|
typename = EnableIfProtosClass<T>> |
|
bool HasExtension( |
|
const T& message, |
|
const ::protos::internal::ExtensionIdentifier<Extendee, Extension>& id) { |
|
return _upb_Message_Getext(message.msg(), id.mini_table_ext()) != nullptr; |
|
} |
|
|
|
template <typename T, typename Extendee, typename Extension, |
|
typename = EnableIfProtosClass<T>> |
|
bool HasExtension( |
|
const Ptr<T>& message, |
|
const ::protos::internal::ExtensionIdentifier<Extendee, Extension>& id) { |
|
return _upb_Message_Getext(message->msg(), id.mini_table_ext()) != nullptr; |
|
} |
|
|
|
template <typename T, typename Extendee, typename Extension, |
|
typename = EnableIfProtosClass<T>, typename = EnableIfMutableProto<T>> |
|
void ClearExtension( |
|
const Ptr<T>& message, |
|
const ::protos::internal::ExtensionIdentifier<Extendee, Extension>& id) { |
|
_upb_Message_ClearExtensionField(message->msg(), id.mini_table_ext()); |
|
} |
|
|
|
template <typename T, typename Extendee, typename Extension, |
|
typename = EnableIfProtosClass<T>> |
|
void ClearExtension( |
|
const T& message, |
|
const ::protos::internal::ExtensionIdentifier<Extendee, Extension>& id) { |
|
_upb_Message_ClearExtensionField(message.msg(), id.mini_table_ext()); |
|
} |
|
|
|
template <typename T, typename Extendee, typename Extension, |
|
typename = EnableIfProtosClass<T>> |
|
absl::Status SetExtension( |
|
const T& message, |
|
const ::protos::internal::ExtensionIdentifier<Extendee, Extension>& id, |
|
Extension& value) { |
|
auto* message_arena = static_cast<upb_Arena*>(message.GetInternalArena()); |
|
upb_Message_Extension* msg_ext = _upb_Message_GetOrCreateExtension( |
|
message.msg(), id.mini_table_ext(), message_arena); |
|
if (!msg_ext) { |
|
return MessageAllocationError(); |
|
} |
|
auto* extension_arena = static_cast<upb_Arena*>(value.GetInternalArena()); |
|
if (message_arena != extension_arena) { |
|
upb_Arena_Fuse(message_arena, extension_arena); |
|
} |
|
msg_ext->data.ptr = value.msg(); |
|
return absl::OkStatus(); |
|
} |
|
|
|
template <typename T, typename Extendee, typename Extension, |
|
typename = EnableIfProtosClass<T>, typename = EnableIfMutableProto<T>> |
|
absl::Status SetExtension( |
|
const Ptr<T>& message, |
|
const ::protos::internal::ExtensionIdentifier<Extendee, Extension>& id, |
|
Extension& value) { |
|
auto* message_arena = static_cast<upb_Arena*>(message->GetInternalArena()); |
|
upb_Message_Extension* msg_ext = _upb_Message_GetOrCreateExtension( |
|
message->msg(), id.mini_table_ext(), message_arena); |
|
if (!msg_ext) { |
|
return MessageAllocationError(); |
|
} |
|
auto* extension_arena = static_cast<upb_Arena*>(message->GetInternalArena()); |
|
if (message_arena != extension_arena) { |
|
upb_Arena_Fuse(message_arena, extension_arena); |
|
} |
|
msg_ext->data.ptr = ::protos::internal::GetInternalMsg(value); |
|
return absl::OkStatus(); |
|
} |
|
|
|
template <typename T, typename Extendee, typename Extension, |
|
typename = EnableIfProtosClass<T>> |
|
absl::StatusOr<Ptr<const Extension>> GetExtension( |
|
const T& message, |
|
const ::protos::internal::ExtensionIdentifier<Extendee, Extension>& id) { |
|
const upb_Message_Extension* ext = |
|
_upb_Message_Getext(message.msg(), id.mini_table_ext()); |
|
if (!ext) { |
|
return ExtensionNotFoundError(id.mini_table_ext()->field.number); |
|
} |
|
return Ptr<const Extension>( |
|
::protos::internal::CreateMessage<Extension>(ext->data.ptr)); |
|
} |
|
|
|
template <typename T, typename Extendee, typename Extension, |
|
typename = EnableIfProtosClass<T>> |
|
absl::StatusOr<Ptr<const Extension>> GetExtension( |
|
const Ptr<T>& message, |
|
const ::protos::internal::ExtensionIdentifier<Extendee, Extension>& id) { |
|
const upb_Message_Extension* ext = |
|
_upb_Message_Getext(message->msg(), id.mini_table_ext()); |
|
if (!ext) { |
|
return ExtensionNotFoundError(id.mini_table_ext()->field.number); |
|
} |
|
return Ptr<const Extension>( |
|
::protos::internal::CreateMessage<Extension>(ext->data.ptr)); |
|
} |
|
|
|
template <typename T> |
|
bool Parse(T& message, absl::string_view bytes) { |
|
_upb_Message_Clear(message.msg(), T::minitable()); |
|
auto* arena = static_cast<upb_Arena*>(message.GetInternalArena()); |
|
return upb_Decode(bytes.data(), bytes.size(), message.msg(), T::minitable(), |
|
/* extreg= */ nullptr, /* options= */ 0, |
|
arena) == kUpb_DecodeStatus_Ok; |
|
} |
|
|
|
template <typename T> |
|
bool Parse(std::unique_ptr<T>& message, absl::string_view bytes) { |
|
_upb_Message_Clear(message->msg(), T::minitable()); |
|
auto* arena = static_cast<upb_Arena*>(message->GetInternalArena()); |
|
return upb_Decode(bytes.data(), bytes.size(), message->msg(), T::minitable(), |
|
/* extreg= */ nullptr, /* options= */ 0, |
|
arena) == kUpb_DecodeStatus_Ok; |
|
} |
|
|
|
template <typename T> |
|
bool Parse(std::shared_ptr<T>& message, absl::string_view bytes) { |
|
_upb_Message_Clear(message->msg(), T::minitable()); |
|
auto* arena = static_cast<upb_Arena*>(message->GetInternalArena()); |
|
return upb_Decode(bytes.data(), bytes.size(), message->msg(), T::minitable(), |
|
/* 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 = static_cast<upb_Arena*>(message.GetInternalArena()); |
|
upb_DecodeStatus status = |
|
upb_Decode(bytes.data(), bytes.size(), message.msg(), T::minitable(), |
|
/* 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 ::protos::ExtensionRegistry& extension_registry, |
|
int options = 0) { |
|
T message; |
|
auto* arena = static_cast<upb_Arena*>(message.GetInternalArena()); |
|
upb_DecodeStatus status = |
|
upb_Decode(bytes.data(), bytes.size(), message.msg(), T::minitable(), |
|
::protos::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 ::protos::internal::Serialize( |
|
::protos::internal::GetInternalMsg(message), T::minitable(), arena.ptr(), |
|
options); |
|
} |
|
|
|
template <typename T> |
|
absl::StatusOr<absl::string_view> Serialize(std::unique_ptr<T>& message, |
|
upb::Arena& arena, |
|
int options = 0) { |
|
return ::protos::internal::Serialize(message->msg(), T::minitable(), |
|
arena.ptr(), options); |
|
} |
|
|
|
template <typename T> |
|
absl::StatusOr<absl::string_view> Serialize(std::shared_ptr<T>& message, |
|
upb::Arena& arena, |
|
int options = 0) { |
|
return ::protos::internal::Serialize(message->msg(), T::minitable(), |
|
arena.ptr(), options); |
|
} |
|
|
|
template <typename T> |
|
absl::StatusOr<absl::string_view> Serialize(Ptr<const T> message, |
|
upb::Arena& arena, |
|
int options = 0) { |
|
return ::protos::internal::Serialize(message->msg(), T::minitable(), |
|
arena.ptr(), options); |
|
} |
|
|
|
} // namespace protos |
|
|
|
#endif // UPB_PROTOS_PROTOS_H_
|
|
|