12#ifndef O2_FRAMEWORK_TABLEBUILDER_H_
13#define O2_FRAMEWORK_TABLEBUILDER_H_
18#include "arrow/type_traits.h"
23#include <arrow/chunked_array.h>
24#include <arrow/status.h>
25#include <arrow/memory_pool.h>
27#include <arrow/type_traits.h>
28#include <arrow/table.h>
29#include <arrow/builder.h>
60template <
typename T,
int N>
65template <
typename T,
int N>
70template <
typename T,
int N>
80#define O2_ARROW_STL_CONVERSION(c_type, ArrowType_) \
82 struct ConversionTraits<c_type> { \
83 using ArrowType = ::arrow::ArrowType_; \
106 template <
typename T>
107 static arrow::Status
appendToList(std::unique_ptr<arrow::FixedSizeListBuilder>& builder, T*
data,
int size = 1)
110 using BuilderType =
typename arrow::TypeTraits<ArrowType>::BuilderType;
111 size_t numElements =
static_cast<const arrow::FixedSizeListType*
>(builder->type().get())->list_size();
113 auto status = builder->AppendValues(
size);
114 auto ValueBuilder =
static_cast<BuilderType*
>(builder->value_builder());
115 status &= ValueBuilder->AppendValues(
data, numElements *
size,
nullptr);
120 template <
typename HolderType,
typename T>
123 return static_cast<typename HolderType::Policy&
>(holder).
append(holder.builder,
value);
126 template <
typename HolderType>
127 static arrow::Status
flush(HolderType& holder)
129 return static_cast<typename HolderType::Policy&
>(holder).
flush(holder.builder);
135 template <
typename HolderType,
typename T>
138 if constexpr (std::is_same_v<
decltype(holder.builder), std::unique_ptr<arrow::FixedSizeListBuilder>>) {
139 return appendToList<T>(holder.builder,
data);
141 return holder.builder->Append(
reinterpret_cast<const uint8_t*
>(
data));
145 template <
typename HolderType,
typename T,
int N>
146 static arrow::Status
append(HolderType& holder, T (&
data)[N])
148 return holder.builder->Append(
reinterpret_cast<const uint8_t*
>(
data));
152 template <
typename HolderType,
typename T,
int N>
153 static arrow::Status
append(HolderType& holder, std::array<T, N>
const&
data)
155 return holder.builder->Append(
reinterpret_cast<const uint8_t*
>(
data.data()));
159 template <
typename HolderType,
typename T>
160 static arrow::Status
append(HolderType& holder, std::vector<T>
const&
data)
163 using ValueBuilderType =
typename arrow::TypeTraits<ArrowType>::BuilderType;
164 auto status = holder.builder->Reserve(
data.size());
165 status &= holder.builder->Append();
166 auto vbuilder =
static_cast<ValueBuilderType*
>(holder.builder->value_builder());
167 status &= vbuilder->AppendValues(
data.begin(),
data.end());
172 template <
typename HolderType,
typename T>
181 template <
typename HolderType,
typename T>
184 return holder.builder->UnsafeAppend(
value);
187 template <
typename HolderType,
typename T>
190 if constexpr (std::is_same_v<
decltype(holder.builder), std::unique_ptr<arrow::FixedSizeListBuilder>>) {
191 auto status = appendToList<T>(holder.builder,
value);
193 return holder.builder->UnsafeAppend(
reinterpret_cast<const uint8_t*
>(
value));
197 template <
typename HolderType,
typename PTR>
198 static arrow::Status
bulkAppend(HolderType& holder,
size_t bulkSize,
const PTR
ptr)
200 return holder.builder->AppendValues(
ptr, bulkSize,
nullptr);
203 template <
typename HolderType,
typename PTR>
207 if (info.
ptr ==
nullptr) {
208 return arrow::Status::OK();
210 if constexpr (std::is_same_v<
decltype(holder.builder), std::unique_ptr<arrow::FixedSizeListBuilder>>) {
211 if (
appendToList<std::remove_pointer_t<
decltype(info.
ptr)>>(holder.builder, info.
ptr, info.
size).ok() ==
false) {
214 return arrow::Status::OK();
217 if (holder.builder->AppendValues(info.
ptr, info.
size,
nullptr).ok() ==
false) {
220 return arrow::Status::OK();
225 template <
typename HolderType,
typename ITERATOR>
226 static arrow::Status
append(HolderType& holder, std::pair<ITERATOR, ITERATOR> ip)
229 using ValueBuilderType =
typename arrow::TypeTraits<ArrowType>::BuilderType;
231 auto status = holder.builder->Append();
232 auto valueBuilder =
reinterpret_cast<ValueBuilderType*
>(holder.builder->value_builder());
233 return status & valueBuilder->AppendValues(&*ip.first, std::distance(ip.first, ip.second));
237 template <
typename HolderType,
typename ITERATOR>
238 static void unsafeAppend(HolderType& holder, std::pair<ITERATOR, ITERATOR> ip)
241 using ValueBuilderType =
typename arrow::TypeTraits<ArrowType>::BuilderType;
243 auto status = holder.builder->Append();
244 auto valueBuilder =
reinterpret_cast<ValueBuilderType*
>(holder.builder->value_builder());
245 status &= valueBuilder->AppendValues(&*ip.first, std::distance(ip.first, ip.second));
247 throw runtime_error(
"Unable to append values to valueBuilder!");
258 using BuilderType =
typename arrow::TypeTraits<ArrowType>::BuilderType;
260 static std::unique_ptr<BuilderType>
make(arrow::MemoryPool* pool)
262 return std::make_unique<BuilderType>(pool);
267 return arrow::TypeTraits<ArrowType>::type_singleton();
272 return builder.Append(
value);
278 return builder.Append(
value);
287 using BuilderType =
typename arrow::TypeTraits<ArrowType>::BuilderType;
289 static std::unique_ptr<BuilderType>
make(arrow::MemoryPool* pool)
291 return std::make_unique<BuilderType>(pool);
296 return arrow::TypeTraits<ArrowType>::type_singleton();
301 return builder.Append(
value);
305template <
typename ITERATOR>
312 using ValueBuilder =
typename arrow::TypeTraits<ValueType>::BuilderType;
314 static std::unique_ptr<BuilderType>
make(arrow::MemoryPool* pool)
316 auto valueBuilder = std::make_shared<ValueBuilder>(pool);
317 return std::make_unique<arrow::ListBuilder>(pool, valueBuilder);
322 return arrow::list(arrow::TypeTraits<ValueType>::type_singleton());
326template <
typename T,
int N>
334 static std::unique_ptr<BuilderType>
make(arrow::MemoryPool* pool)
336 std::unique_ptr<arrow::ArrayBuilder> valueBuilder;
338 arrow::MakeBuilder(pool, arrow::TypeTraits<ElementType>::type_singleton(), &valueBuilder);
339 return std::make_unique<BuilderType>(pool, std::move(valueBuilder), N);
344 return arrow::fixed_size_list(arrow::TypeTraits<ElementType>::type_singleton(), N);
348template <
typename T,
int N>
355 static std::unique_ptr<BuilderType>
make(arrow::MemoryPool* pool)
357 std::unique_ptr<arrow::ArrayBuilder> valueBuilder;
359 arrow::MakeBuilder(pool, arrow::TypeTraits<ElementType>::type_singleton(), &valueBuilder);
360 return std::make_unique<BuilderType>(pool, std::move(valueBuilder), N);
365 return arrow::fixed_size_list(arrow::TypeTraits<ElementType>::type_singleton(), N);
369template <
typename T,
int N>
376 static std::unique_ptr<BuilderType>
make(arrow::MemoryPool* pool)
378 std::unique_ptr<arrow::ArrayBuilder> valueBuilder;
380 arrow::MakeBuilder(pool, arrow::TypeTraits<ElementType>::type_singleton(), &valueBuilder);
381 return std::make_unique<BuilderType>(pool, std::move(valueBuilder), N);
386 return arrow::fixed_size_list(arrow::TypeTraits<ElementType>::type_singleton(), N);
397 static std::unique_ptr<BuilderType>
make(arrow::MemoryPool* pool)
399 std::unique_ptr<arrow::ArrayBuilder> valueBuilder;
401 arrow::MakeBuilder(pool, arrow::TypeTraits<ElementType>::type_singleton(), &valueBuilder);
402 return std::make_unique<BuilderType>(pool, std::move(valueBuilder));
407 return arrow::list(arrow::TypeTraits<ElementType>::type_singleton());
411template <
typename... ARGS>
414 return std::make_tuple(std::make_unique<ARGS>()...);
420 using BuilderType =
typename arrow::TypeTraits<ArrowType>::BuilderType;
425template <
typename ITERATOR>
433template <
typename T,
int N>
447 template <
typename BUILDER>
450 return builder->Append(
value);
453 template <
typename BUILDER>
456 return arrow::Status::OK();
464 template <
typename BUILDER>
472 return arrow::Status::OK();
475 template <
typename BUILDER>
476 arrow::Status
flush(BUILDER& builder)
481 return arrow::Status::OK();
487template <
size_t I,
typename T,
typename P>
492 using BuilderType =
typename arrow::TypeTraits<ArrowType>::BuilderType;
498 auto s =
builder->Reserve(nRows);
509 template <
typename... ARGS,
size_t NCOLUMNS>
515 template <
typename... ARGS,
size_t NCOLUMNS =
sizeof...(ARGS)>
516 static std::vector<std::shared_ptr<arrow::Field>>
makeFields(std::array<char const*, NCOLUMNS>
const& names)
518 char const*
const* names_ptr = names.data();
524 template <
typename...
Ts,
typename VALUES>
533 template <
typename...
Ts,
typename VALUES>
539 template <
typename...
Ts,
typename PTRS>
540 static bool bulkAppend(std::tuple<Ts...>& holders,
size_t bulkSize, PTRS ptrs)
546 template <
typename...
Ts,
typename INFOS>
553 template <
typename...
Ts>
554 static bool finalize(std::vector<std::shared_ptr<arrow::Array>>&
arrays, std::tuple<Ts...>& holders)
556 return (
finalize(
arrays[Ts::index], std::get<Ts::index>(holders)) && ...);
559 template <
typename HOLDER>
566template <
typename... ARGS>
573concept BulkInsertable = (std::integral<std::decay<T>> && !std::same_as<bool, std::decay_t<T>>);
589 using type = std::decay_t<T>;
592 return std::make_tuple(p0,
p1,
p2, p3);
595 return std::make_tuple(p0,
p1,
p2);
598 return std::make_tuple(p0,
p1);
601 return std::make_tuple(p0);
603 return std::make_tuple();
607template <
typename... ARGS>
610 return []<std::size_t... Is>(std::index_sequence<Is...>) {
612 }(std::make_index_sequence<
sizeof...(ARGS)>{});
615template <
typename... ARGS>
618 return [pool, nRows]<std::size_t... Is>(std::index_sequence<Is...>) {
620 }(std::make_index_sequence<
sizeof...(ARGS)>{});
623template <
typename... ARGS>
627concept ShouldNotDeconstruct = std::is_bounded_array_v<T> || std::is_arithmetic_v<T> || framework::is_base_of_template_v<std::vector, T>;
636 template <
typename... ARGS>
637 using HoldersTuple =
typename std::tuple<BuilderHolder<0, ARGS, typename InsertionTrait<ARGS>::Policy>...>;
639 template <
typename... ARGS>
644 template <
typename... ARGS>
647 return (HoldersTupleIndexed<ARGS...>*)mHolders;
650 void validate()
const;
652 template <
typename... ARGS,
size_t I =
sizeof...(ARGS)>
653 auto makeBuilders(std::array<char const*, I>
const& columnNames,
size_t nRows)
655 mSchema = std::make_shared<arrow::Schema>(TableBuilderHelpers::makeFields<ARGS...>(columnNames));
657 mHolders =
makeHolders<ARGS...>(mMemoryPool, nRows);
658 mFinalizer = [](std::vector<std::shared_ptr<arrow::Array>>&
arrays,
void* holders) ->
bool {
661 mDestructor = [](
void* holders)
mutable ->
void {
662 delete (HoldersTupleIndexed<ARGS...>*)holders;
667 template <
typename ARG0,
typename... ARGS>
675 template <
typename ARG0,
typename... ARGS>
679 return 1 +
sizeof...(ARGS);
692 mDestructor(mHolders);
697 template <
typename ARG0,
typename... ARGS>
699 auto persist(std::array<
char const*,
sizeof...(ARGS) + 1>
const& columnNames)
704 persister(slot, std::forward_as_tuple(arg, args...));
710 template <
typename ARG0,
typename... ARGS>
712 auto persist(std::array<
char const*, countColumns<ARG0, ARGS...>()>
const& columnNames)
715 auto persister =
persistTuple(argsPack_t{}, columnNames);
716 return [persister = persister](
unsigned int slot, ARG0
const& obj) ->
void {
723 template <
typename... ARGS>
726 constexpr int nColumns =
sizeof...(ARGS);
728 mArrays.resize(nColumns);
729 makeBuilders<ARGS...>(columnNames, 10);
732 using FillTuple = std::tuple<typename BuilderMaker<ARGS>::FillType...>;
733 return [holders = mHolders](
unsigned int , FillTuple
const& t) ->
void {
735 if (status ==
false) {
743 template <
typename T>
746 return [
this]<
typename... Cs>(
pack<Cs...>) {
747 return this->
template persist<
typename Cs::type...>({Cs::columnLabel()...});
748 }(
typename T::table_t::persistent_columns_t{});
751 template <
typename... Cs>
754 return this->
template persist<
typename Cs::type...>({Cs::columnLabel()...});
757 template <
typename T,
typename E>
760 return [
this]<
typename... Cs>(
pack<Cs...>) {
761 return this->
template persist<E>({Cs::columnLabel()...});
762 }(
typename T::table_t::persistent_columns_t{});
765 template <
typename... ARGS,
size_t NCOLUMNS =
sizeof...(ARGS)>
768 constexpr size_t nColumns = NCOLUMNS;
770 mArrays.resize(nColumns);
771 makeBuilders<ARGS...>(columnNames, nRows);
779 template <
typename... ARGS,
size_t NCOLUMNS =
sizeof...(ARGS)>
780 auto bulkPersist(std::array<char const*, NCOLUMNS>
const& columnNames,
size_t nRows)
784 mArrays.resize(NCOLUMNS);
785 makeBuilders<ARGS...>(columnNames, nRows);
792 template <
typename... ARGS,
size_t NCOLUMNS =
sizeof...(ARGS)>
796 mArrays.resize(NCOLUMNS);
797 makeBuilders<ARGS...>(columnNames, nRows);
805 template <
typename...
Ts>
808 return (std::get<Ts::index>(holders).builder->Reserve(s).ok() && ...);
811 template <
typename... ARGS>
819 std::shared_ptr<arrow::Table>
finalize();
822 bool (*mFinalizer)(std::vector<std::shared_ptr<arrow::Array>>&
arrays,
void* holders);
823 void (*mDestructor)(
void* holders);
825 arrow::MemoryPool* mMemoryPool;
826 std::shared_ptr<arrow::Schema> mSchema;
827 std::vector<std::shared_ptr<arrow::Array>> mArrays;
834 [[maybe_unused]]
auto writer =
b.
cursor<T>();
839template <soa::TableRef R>
844 b.setLabel(aod::label<R>());
848template <
typename... Cs>
852 [[maybe_unused]]
auto writer =
b.
cursor(p);
857std::shared_ptr<arrow::Table>
spawnerHelper(std::shared_ptr<arrow::Table>
const& fullTable, std::shared_ptr<arrow::Schema> newSchema,
size_t nColumns,
861template <aod::is_aod_hash D>
862auto spawner(std::vector<std::shared_ptr<arrow::Table>>&& tables,
const char*
name)
866 if (fullTable->num_rows() == 0) {
870 static auto new_schema = std::make_shared<arrow::Schema>(fields);
873 return {{std::move(C::Projector())...}};
875 (expression_pack_t{});
880template <aod::is_aod_hash D>
881auto spawner(std::shared_ptr<arrow::Table>
const& fullTable,
const char*
name)
884 if (fullTable->num_rows() == 0) {
888 static auto new_schema = std::make_shared<arrow::Schema>(fields);
891 return {{std::move(C::Projector())...}};
893 (expression_pack_t{});
911template <
typename... C>
915 if (fullTable->num_rows() == 0) {
919 static auto new_schema = std::make_shared<arrow::Schema>(fields);
921 return spawnerHelper(fullTable, new_schema,
sizeof...(C), projectors.data(), fields,
name);
924template <
typename... T>
constexpr int p1()
constexpr to accelerate the coordinates changing
#define O2_ARROW_STL_CONVERSION(c_type, ArrowType_)
auto cursor(framework::pack< Cs... >)
auto persist(std::array< char const *, countColumns< ARG0, ARGS... >()> const &columnNames)
auto persistTuple(framework::pack< ARGS... >, std::array< char const *, sizeof...(ARGS)> const &columnNames)
Same a the above, but use a tuple to persist stuff.
void setLabel(const char *label)
static constexpr int countColumns()
auto bulkPersistChunked(std::array< char const *, NCOLUMNS > const &columnNames, size_t nRows)
auto bulkPersist(std::array< char const *, NCOLUMNS > const &columnNames, size_t nRows)
void extracted(bool &status)
Actually creates the arrow::Table from the builders.
static constexpr int countColumns()
auto reserve(o2::framework::pack< ARGS... > &&, int s)
TableBuilder(arrow::MemoryPool *pool=arrow::default_memory_pool())
auto reserveArrays(std::tuple< Ts... > &holders, int s)
Reserve method to expand the columns as needed.
auto persist(std::array< char const *, sizeof...(ARGS)+1 > const &columnNames)
std::shared_ptr< arrow::Table > finalize()
auto preallocatedPersist(std::array< char const *, NCOLUMNS > const &columnNames, int nRows)
GLuint const GLchar * name
GLboolean GLboolean GLboolean b
GLsizei const GLfloat * value
GLint GLint GLsizei GLint GLenum GLenum type
GLenum GLsizei GLsizei GLint * values
GLuint GLsizei const GLchar * label
typedef void(APIENTRYP PFNGLCULLFACEPROC)(GLenum mode)
Defining PrimaryVertex explicitly as messageable.
void addLabelToSchema(std::shared_ptr< arrow::Schema > &schema, const char *label)
RuntimeErrorRef runtime_error(const char *)
constexpr auto makeHolderTypes()
auto constexpr to_tuple(T &&object) noexcept
std::shared_ptr< arrow::Table > spawnerHelper(std::shared_ptr< arrow::Table > const &fullTable, std::shared_ptr< arrow::Schema > newSchema, size_t nColumns, expressions::Projector *projectors, std::vector< std::shared_ptr< arrow::Field > > const &fields, const char *name)
constexpr auto tuple_to_pack(std::tuple< ARGS... > &&)
auto spawner(std::vector< std::shared_ptr< arrow::Table > > &&tables, const char *name)
Expression-based column generator to materialize columns.
auto makeHolders(arrow::MemoryPool *pool, size_t nRows)
constexpr std::size_t pack_size(pack< Ts... > const &)
template function to determine number of types in a pack
decltype(makeHolderTypes< ARGS... >()) IndexedHoldersTuple
RuntimeErrorRef runtime_error_f(const char *,...)
std::tuple< typename T::iterator... > iterator_tuple_t
auto createFieldsFromColumns(framework::pack< C... >)
Defining DataPointCompositeObject explicitly as copiable.
BuilderHolder(arrow::MemoryPool *pool, size_t nRows=0)
std::unique_ptr< BuilderType > builder
typename arrow::TypeTraits< ArrowType >::BuilderType BuilderType
typename detail::ConversionTraits< T >::ArrowType ArrowType
typename detail::ConversionTraits< T >::ArrowType ElementType
static std::unique_ptr< BuilderType > make(arrow::MemoryPool *pool)
arrow::FixedSizeListBuilder BuilderType
arrow::FixedSizeListType ArrowType
static std::shared_ptr< arrow::DataType > make_datatype()
static std::unique_ptr< BuilderType > make(arrow::MemoryPool *pool)
typename detail::ConversionTraits< T >::ArrowType ElementType
arrow::FixedSizeListType ArrowType
static std::shared_ptr< arrow::DataType > make_datatype()
arrow::FixedSizeListBuilder BuilderType
typename arrow::TypeTraits< ArrowType >::BuilderType BuilderType
static arrow::Status append(BuilderType &builder, bool value)
static std::unique_ptr< BuilderType > make(arrow::MemoryPool *pool)
static std::shared_ptr< arrow::DataType > make_datatype()
typename detail::ConversionTraits< bool >::ArrowType ArrowType
typename detail::ConversionTraits< T >::ArrowType ElementType
static std::unique_ptr< BuilderType > make(arrow::MemoryPool *pool)
arrow::FixedSizeListType ArrowType
arrow::FixedSizeListBuilder BuilderType
static std::shared_ptr< arrow::DataType > make_datatype()
static std::unique_ptr< BuilderType > make(arrow::MemoryPool *pool)
std::pair< ITERATOR, ITERATOR > FillType
arrow::ListType ArrowType
arrow::ListBuilder BuilderType
typename detail::ConversionTraits< typename ITERATOR::value_type >::ArrowType ValueType
typename ITERATOR::value_type STLValueType
typename arrow::TypeTraits< ValueType >::BuilderType ValueBuilder
static std::shared_ptr< arrow::DataType > make_datatype()
arrow::ListType ArrowType
static std::unique_ptr< BuilderType > make(arrow::MemoryPool *pool)
typename detail::ConversionTraits< T >::ArrowType ElementType
arrow::ListBuilder BuilderType
std::vector< T > FillType
static std::shared_ptr< arrow::DataType > make_datatype()
typename arrow::TypeTraits< ArrowType >::BuilderType BuilderType
static arrow::Status append(BuilderType &builder, std::array< T, N > &value)
static arrow::Status append(BuilderType &builder, T value)
typename detail::ConversionTraits< T >::ArrowType ArrowType
static std::unique_ptr< BuilderType > make(arrow::MemoryPool *pool)
static std::shared_ptr< arrow::DataType > make_datatype()
arrow::FixedSizeListBuilder BuilderType
arrow::FixedSizeListType ArrowType
arrow::ListType ArrowType
arrow::ListBuilder BuilderType
arrow::ListType ArrowType
arrow::ListBuilder BuilderType
typename arrow::TypeTraits< ArrowType >::BuilderType BuilderType
typename detail::ConversionTraits< T >::ArrowType ArrowType
static arrow::Status bulkAppend(HolderType &holder, size_t bulkSize, const PTR ptr)
static arrow::Status append(HolderType &holder, std::vector< T > const &data)
Appender for the vector case.
static arrow::Status appendToList(std::unique_ptr< arrow::FixedSizeListBuilder > &builder, T *data, int size=1)
static arrow::Status flush(HolderType &holder)
static arrow::Status append(HolderType &holder, T value)
static arrow::Status append(HolderType &holder, T(&data)[N])
Appender for the array case.
static arrow::Status append(HolderType &holder, T *data)
static void unsafeAppend(HolderType &holder, std::vector< T > const &value)
static arrow::Status append(HolderType &holder, std::pair< ITERATOR, ITERATOR > ip)
static void unsafeAppend(HolderType &holder, T *value)
static void unsafeAppend(HolderType &holder, std::pair< ITERATOR, ITERATOR > ip)
static arrow::Status append(HolderType &holder, std::array< T, N > const &data)
Appender for the array case.
static arrow::Status bulkAppendChunked(HolderType &holder, BulkInfo< PTR > info)
static void unsafeAppend(HolderType &holder, T value)
static constexpr int CHUNK_SIZE
arrow::Status append(BUILDER &builder, T value)
arrow::Status flush(BUILDER &builder)
arrow::Status append(BUILDER &builder, T value)
arrow::Status flush(BUILDER &)
static consteval DirectInsertion< T > policy()
decltype(policy()) Policy
static consteval CachedInsertion< T > policy()
static bool bulkAppendChunked(std::tuple< Ts... > &holders, INFOS infos)
Return true if all columns are done.
static bool finalize(std::shared_ptr< arrow::Array > &array, HOLDER &holder)
static bool append(std::tuple< Ts... > &holders, VALUES &&values)
Invokes the append method for each entry in the tuple.
static std::array< arrow::DataType, NCOLUMNS > makeArrowColumnTypes()
static bool finalize(std::vector< std::shared_ptr< arrow::Array > > &arrays, std::tuple< Ts... > &holders)
Invokes the append method for each entry in the tuple.
static bool bulkAppend(std::tuple< Ts... > &holders, size_t bulkSize, PTRS ptrs)
static void unsafeAppend(std::tuple< Ts... > &holders, VALUES &&values)
static std::vector< std::shared_ptr< arrow::Field > > makeFields(std::array< char const *, NCOLUMNS > const &names)
::arrow::FixedSizeListType ArrowType
::arrow::FixedSizeListType ArrowType
::arrow::FixedSizeListType ArrowType
::arrow::ListType ArrowType
FIXME: adapt type conversion to new arrow.
A struct, containing the root of the expression tree.
static std::shared_ptr< arrow::Table > joinTables(std::vector< std::shared_ptr< arrow::Table > > &&tables)