3#include <ydb-cpp-sdk/client/value/value.h>
10#include <userver/utils/assert.hpp>
11#include <userver/utils/meta.hpp>
13#include <userver/ydb/io/traits.hpp>
15USERVER_NAMESPACE_BEGIN
22class ParseItemsIterator final {
24 using difference_type = std::ptrdiff_t;
27 using iterator_category = std::input_iterator_tag;
29 ParseItemsIterator() =
default;
31 ParseItemsIterator(NYdb::TValueParser& parser,
const ParseContext& context) : parser_(&parser), context_(&context) {
36 ParseItemsIterator(
const ParseItemsIterator&) =
default;
37 ParseItemsIterator& operator=(
const ParseItemsIterator&) =
default;
41 return ydb::Parse<T>(*parser_, *context_);
44 ParseItemsIterator& operator++() {
46 if (!parser_->TryNextListItem()) {
52 bool operator==(
const ParseItemsIterator& other)
const noexcept {
53 UASSERT(parser_ ==
nullptr || other.parser_ ==
nullptr);
54 return parser_ == other.parser_;
58 NYdb::TValueParser* parser_{
nullptr};
59 const ParseContext* context_{
nullptr};
66using InsertRow = std::vector<InsertColumn>;
70struct ValueTraits<T, std::enable_if_t<meta::kIsRange<T> && !meta::kIsMap<T>>> {
71 using ValueType = meta::RangeValueType<T>;
73 static T Parse(NYdb::TValueParser& parser,
const ParseContext& context) {
75#if __cpp_lib_ranges_to_container >= 202202L
77 std::ranges::subrange{
78 impl::ParseItemsIterator<ValueType>{parser, context},
79 impl::ParseItemsIterator<ValueType>{},
83 T result(impl::ParseItemsIterator<ValueType>{parser, context}, impl::ParseItemsIterator<ValueType>{});
89 template <
typename Builder,
typename U =
const T&>
90 static void Write(NYdb::TValueBuilderBase<Builder>& builder, U&& value) {
91 bool list_opened =
false;
92 for (
auto&& item : value) {
97 builder.AddListItem();
98 ydb::Write(builder, item);
102 }
else if constexpr (std::is_same_v<ValueType, InsertRow>) {
107 builder.EmptyList(ValueTraits<ValueType>::MakeType());
111 static NYdb::TType MakeType() {
112 NYdb::TTypeBuilder builder;
113 builder.List(ValueTraits<ValueType>::MakeType());
114 return builder.Build();