8#include <userver/utils/assert.hpp>
10#include <userver/storages/clickhouse/impl/block_wrapper_fwd.hpp>
11#include <userver/storages/clickhouse/io/impl/validate.hpp>
13#include <userver/storages/clickhouse/io/columns/column_wrapper.hpp>
14#include <userver/storages/clickhouse/io/columns/common_columns.hpp>
15#include <userver/storages/clickhouse/io/columns/nullable_column.hpp>
17USERVER_NAMESPACE_BEGIN
21class InsertionRequest
final {
23 InsertionRequest(
const std::string& table_name,
24 const std::vector<std::string_view>& column_names);
25 InsertionRequest(InsertionRequest&&)
noexcept;
29 static InsertionRequest Create(
30 const std::string& table_name,
31 const std::vector<std::string_view>& column_names,
const T& data);
33 template <
typename Container>
34 static InsertionRequest CreateFromRows(
35 const std::string& table_name,
36 const std::vector<std::string_view>& column_names,
const Container& data);
38 const std::string& GetTableName()
const;
40 const impl::BlockWrapper& GetBlock()
const;
43 template <
typename MappedType>
44 class ColumnsMapper
final {
46 ColumnsMapper(impl::BlockWrapper& block,
47 const std::vector<std::string_view>& column_names)
48 : block_{block}, column_names_{column_names} {}
50 template <
typename Field, size_t Index>
51 void operator()(
const Field& field,
52 std::integral_constant<size_t, Index> i) {
53 using ColumnType = std::tuple_element_t<Index, MappedType>;
54 static_assert(std::is_same_v<Field,
typename ColumnType::container_type>);
56 io::columns::AppendWrappedColumn(block_, ColumnType::Serialize(field),
61 impl::BlockWrapper& block_;
62 const std::vector<std::string_view>& column_names_;
65 template <
typename MappedType,
typename Container>
66 class RowsMapper
final {
68 RowsMapper(impl::BlockWrapper& block,
69 const std::vector<std::string_view>& column_names,
70 const Container& data)
71 : block_{block}, column_names_{column_names}, data_{data} {}
73 template <
typename Field, size_t Index>
74 void operator()(
const Field&, std::integral_constant<size_t, Index> i) {
75 using ColumnType = std::tuple_element_t<Index, MappedType>;
76 static_assert(std::is_same_v<Field,
typename ColumnType::cpp_type>);
78 std::vector<Field> column_data;
79 column_data.reserve(data_.size());
80 for (
const auto& row : data_) {
81 column_data.emplace_back(boost::pfr::get<Index>(row));
84 io::columns::AppendWrappedColumn(
85 block_, ColumnType::Serialize(column_data), column_names_[i], i);
89 impl::BlockWrapper& block_;
90 const std::vector<std::string_view>& column_names_;
91 const Container& data_;
94 const std::string& table_name_;
95 const std::vector<std::string_view>& column_names_;
97 std::unique_ptr<impl::BlockWrapper> block_;
101InsertionRequest InsertionRequest::Create(
102 const std::string& table_name,
103 const std::vector<std::string_view>& column_names,
const T& data) {
104 io::impl::ValidateColumnsMapping(data);
105 io::impl::ValidateRowsCount(data);
107 io::impl::ValidateColumnsCount<T>(column_names.size());
109 InsertionRequest request{table_name, column_names};
110 using MappedType =
typename io::CppToClickhouse<T>::mapped_type;
111 auto mapper = InsertionRequest::ColumnsMapper<MappedType>{
112 *request.block_, request.column_names_};
114 boost::pfr::for_each_field(data, mapper);
118template <
typename Container>
119InsertionRequest InsertionRequest::CreateFromRows(
120 const std::string& table_name,
121 const std::vector<std::string_view>& column_names,
const Container& data) {
122 using T =
typename Container::value_type;
123 io::impl::CommonValidateMapping<T>();
125 io::impl::ValidateColumnsCount<T>(column_names.size());
127 UINVARIANT(!data.empty(),
"An attempt to insert empty chunk of data");
129 InsertionRequest request{table_name, column_names};
130 using MappedType =
typename io::CppToClickhouse<T>::mapped_type;
131 auto mapper = InsertionRequest::RowsMapper<MappedType, Container>{
132 *request.block_, request.column_names_, data};
134 boost::pfr::for_each_field(data.front(), mapper);