/* * Copyright (c) Meta Platforms, Inc. and affiliates. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #pragma once #include #include #include #include #include #include #include #include #include #include #include #include namespace apache::thrift::test { namespace basic { class MyStruct; enum class AdaptedEnum { Zero = 0, One, Two, }; } // namespace basic template struct Wrapper { T value; Wrapper& operator=(const T&) = delete; bool operator==(const Wrapper& other) const { return value == other.value; } bool operator<(const Wrapper& other) const { return value < other.value; } }; struct TemplatedTestAdapter { template static Wrapper fromThrift(T value) { return {value}; } template static auto&& toThrift(Wrapper&& wrapper) { return std::forward(wrapper).value; } }; struct AdaptTestMsAdapter { static std::chrono::milliseconds fromThrift(int64_t ms) { return std::chrono::milliseconds{ms}; } static int64_t toThrift(std::chrono::milliseconds duration) { return duration.count(); } }; struct AdapterEqualsStringAdapter { std::string value; static std::string fromThrift(std::string val) { return val; } static const std::string& toThrift(const std::string& str) { return str; } static bool equal(const std::string& lhs, const std::string& rhs) { return lhs != rhs; } }; struct AdaptedEqualsString { std::string val; bool operator==(const AdaptedEqualsString& other) const { return val != other.val; } }; struct AdaptedEqualsStringAdapter { AdaptedEqualsString val; static AdaptedEqualsString fromThrift(std::string&& val) { return AdaptedEqualsString{std::move(val)}; } static std::string toThrift(const AdaptedEqualsString& val) { return val.val; } }; struct AdapterComparisonStringAdapter { std::string value; static std::string fromThrift(std::string&& val) { return std::move(val); } static const std::string& toThrift(const std::string& str) { return str; } static bool less(const std::string& lhs, const std::string& rhs) { return lhs > rhs; } }; struct AdaptedComparisonString { std::string val; bool operator<(const AdaptedComparisonString& other) const { return val > other.val; } }; struct AdaptedComparisonStringAdapter { AdaptedComparisonString val; static AdaptedComparisonString fromThrift(std::string&& val) { return AdaptedComparisonString{std::move(val)}; } static std::string toThrift(const AdaptedComparisonString& val) { return val.val; } }; struct Num { int64_t val = 13; private: friend bool operator==(const Num& lhs, const Num& rhs) { return lhs.val == rhs.val; } friend bool operator<(const Num& lhs, const Num& rhs) { return lhs.val < rhs.val; } }; struct String { std::string val; }; template struct Indirection { FBTHRIFT_CPP_DEFINE_MEMBER_INDIRECTION_FN(val); T val; }; using IndirectionString = Indirection; using IndirectionIOBuf = Indirection; struct OverloadedAdapter { static Num fromThrift(int64_t val) { return Num{val}; } static String fromThrift(std::string&& val) { return String{std::move(val)}; } static int64_t toThrift(const Num& num) { return num.val; } static const std::string& toThrift(const String& str) { return str.val; } }; struct CustomProtocolAdapter { static Num fromThrift(const folly::IOBuf& data) { if (data.length() < sizeof(int64_t)) { throw std::invalid_argument("CustomProtocolAdapter parse error"); } return {*reinterpret_cast(data.data())}; } static folly::IOBuf toThrift(const Num& num) { return folly::IOBuf::wrapBufferAsValue(&num.val, sizeof(int64_t)); } }; // TODO(afuller): Move this to a shared location. template struct IndirectionAdapter { template FOLLY_ERASE static constexpr AdaptedT fromThrift(ThriftT&& value) { AdaptedT adapted; toThrift(adapted) = std::forward(value); return adapted; } FOLLY_ERASE static constexpr decltype(auto) toThrift(AdaptedT& adapted) noexcept( noexcept(::apache::thrift::apply_indirection(adapted))) { return ::apache::thrift::apply_indirection(adapted); } }; template struct AdaptedWithContext { T value = {}; int16_t fieldId = 0; std::string* meta = nullptr; AdaptedWithContext() = default; explicit AdaptedWithContext(T value_) : value(value_) {} AdaptedWithContext(T value_, int16_t fieldId_, std::string* meta_) : value(value_), fieldId(fieldId_), meta(meta_) {} }; template inline bool operator==( const AdaptedWithContext& lhs, const AdaptedWithContext& rhs) { return lhs.value == rhs.value; } template inline bool operator<( const AdaptedWithContext& lhs, const AdaptedWithContext& rhs) { return lhs.value < rhs.value; } struct TemplatedTestFieldAdapter { template static void construct( AdaptedWithContext& field, apache::thrift::FieldContext&&) { field.fieldId = apache::thrift::FieldContext::kFieldId; } template static AdaptedWithContext fromThriftField( T value, apache::thrift::FieldContext&&) { return { value, apache::thrift::FieldContext::kFieldId, nullptr, }; } template static T toThrift(const AdaptedWithContext& adapted) { return adapted.value; } }; struct AdapterWithContext { template static void construct( AdaptedWithContext& field, apache::thrift::FieldContext&& ctx) { field.fieldId = apache::thrift::FieldContext::kFieldId; field.meta = &*ctx.object.meta_ref(); } template static AdaptedWithContext fromThriftField( T value, apache::thrift::FieldContext&& ctx) { return { value, apache::thrift::FieldContext::kFieldId, &*ctx.object.meta_ref()}; } template static T toThrift(const AdaptedWithContext& adapted) { return adapted.value; } }; struct AdapterWithContextOptimized { template static void construct( AdaptedWithContext& field, apache::thrift::FieldContext&& ctx) { field.fieldId = apache::thrift::FieldContext::kFieldId; field.meta = &*ctx.object.meta_ref(); } template static AdaptedWithContext fromThriftField( T value, apache::thrift::FieldContext&& ctx) { return { value, apache::thrift::FieldContext::kFieldId, &*ctx.object.meta_ref()}; } template static T toThrift(const AdaptedWithContext& adapted) { return adapted.value; } template static void clear(AdaptedWithContext& field) { field.value = {}; } template static bool isEmpty(const AdaptedWithContext&) { return false; } }; template struct IdentityAdapter { static T fromThrift(T i) { return std::move(i); } static const T& toThrift(const T& i) { return i; } }; template < class T, class U = std::remove_reference_t().field())>> struct TaggedWrapper; struct MemberAccessAdapter { template static TaggedWrapper fromThrift(const T& t) { return {*t.field()}; } template static T toThrift(const TaggedWrapper& w) { T ret; ret.field() = w.val; return ret; } }; template struct TaggedWrapper { U val; bool operator==(const TaggedWrapper& other) const { return val == other.val; } bool operator<(const TaggedWrapper& other) const { return val < other.val; } }; struct MoveOnlyAdapter { template static std::unique_ptr fromThrift(const T& i) { return std::make_unique(i); } template static const T& toThrift(const std::unique_ptr& i) { return *i; } }; template struct CountingAdapter { static inline int count = 0; static T fromThrift(T i) { return i; } static T fromThriftField(T i) { return i; } static T toThrift(T i) { ++count; return i; } template < bool ZC, typename Tag, typename Protocol, bool Enable = hasSerializedSize> static std::enable_if_t serializedSize(Protocol& prot, T) { return prot.serializedSizeI64(); } }; struct SerializedSizeAdapter { template static T fromThrift(T i) { return i; } template static T toThrift(T i) { return i; } static inline uint32_t mockSize = 0; static inline uint32_t mockSizeZeroCopy = 0; template static uint32_t serializedSize(Protocol&, T) { static_assert(std::is_same_v, T>); return ZC ? mockSizeZeroCopy : mockSize; } }; struct EncodeAdapter { static Num fromThrift(int64_t val) { ADD_FAILURE() << "Adapter::decode should be called instead of deserializing with fromThrift."; return Num{val}; } static int64_t toThrift(const Num& num) { ADD_FAILURE() << "Adapter::encode should be called instead of serializing with toThrift."; return num.val; } template static uint32_t encode(Protocol& prot_, const Num& num) { return op::encode(prot_, num.val); } template static void decode(Protocol& prot_, Num& num) { return op::decode(prot_, num.val); } }; struct InPlaceDeserializationAdapter { template static Wrapper fromThrift(T value) { ADD_FAILURE() << "Both serialization and in-place deserialization use toThrift instead."; return {value}; } template static auto&& toThrift(Wrapper&& wrapper) { return std::forward(wrapper).value; } }; struct NoEncodeAdapter { static Num fromThrift(int64_t val) { return Num{val}; } static int64_t toThrift(const Num& num) { return num.val; } }; struct EncodeFieldAdapter { template static AdaptedWithContext fromThriftField( T value, apache::thrift::FieldContext&&) { ADD_FAILURE() << "Adapter::decode should be called instead of deserializing with fromThriftField."; return { value, apache::thrift::FieldContext::kFieldId, nullptr, }; } template static T toThrift(const AdaptedWithContext& adapted) { ADD_FAILURE() << "Adapter::encode should be called instead of serializing with toThrift."; return adapted.value; } template < typename Tag, typename Protocol, typename T, typename Struct, int16_t FieldId> static uint32_t encode( Protocol& prot_, const AdaptedWithContext& adapted) { return op::encode(prot_, adapted.value); } template < typename Tag, typename Protocol, typename T, typename Struct, int16_t FieldId> static void decode( Protocol& prot_, AdaptedWithContext& adapted) { return op::decode(prot_, adapted.value); } }; struct EncodeTemplatedTestAdapter { template static Wrapper fromThrift(T value) { ADD_FAILURE() << "Adapter::decode should be called instead of deserializing with fromThrift."; return {value}; } template static auto&& toThrift(Wrapper&& wrapper) { ADD_FAILURE() << "Adapter::encode should be called instead of serializing with toThrift."; return std::forward(wrapper).value; } template static uint32_t encode(Protocol& prot_, const Wrapper& adapted) { return op::encode(prot_, adapted.value); } template static void decode(Protocol& prot_, Wrapper& adapted) { return op::decode(prot_, adapted.value); } }; template struct VariableWrapper { VariableWrapper(const VariableWrapper&) = delete; VariableWrapper(const T& v) : value{v} {} VariableWrapper(const T& v, std::string n, std::string u) : value{v}, name{n}, uri{u} {} T value; std::string name = ""; std::string uri = ""; }; struct VariableAdapter { // This is necessary as type hint, since adapted_t doesn't support forwarding // annotation template static VariableWrapper fromThrift(T value) { return {value}; } template static VariableWrapper fromThrift( T value, Anno annotation, std::string uri) { return {value, *annotation.name(), uri}; } }; template struct NonComparableWrapper { T value; }; struct NonComparableWrapperAdapter { template static NonComparableWrapper fromThrift(T value) { return {value}; } template static auto&& toThrift(Wrapper&& wrapper) { return std::forward(wrapper).value; } }; class Timestamp { public: FBTHRIFT_CPP_DEFINE_MEMBER_INDIRECTION_FN(value()); uint64_t value() const { return v_; } uint64_t& value() { return v_; } bool operator==(Timestamp that) const { return v_ == that.v_; } bool operator<(Timestamp that) const { return v_ < that.v_; } private: uint64_t v_{0}; }; struct I32ToStringAdapter { static std::string fromThrift(int32_t val) { return folly::to(val); } static int32_t toThrift(const std::string& val) { return folly::to(val); } }; template struct WrappedMyStruct : type::detail::Wrap { // Limit the type to MyStruct, which is defined as // // class MyStruct { 1: i64 field1; } static_assert(std::is_same_v); template uint32_t encode(Protocol& prot) const { encodeCount++; T t; t.field1() = 10; return op::encode>(prot, t); } template void decode(Protocol& prot) { decodeCount++; T t; op::decode>(prot, t); type::detail::Wrap::toThrift().field1() = 20; } mutable size_t encodeCount = 0, decodeCount = 0; }; } // namespace apache::thrift::test