#include #include #include #include #include namespace DB { namespace { /** byteSize() - get the value size in number of bytes for accounting purposes. */ class FunctionByteSize : public IFunction { public: static constexpr auto name = "byteSize"; static FunctionPtr create(ContextConstPtr) { return std::make_shared(); } String getName() const override { return name; } bool useDefaultImplementationForNulls() const override { return false; } bool useDefaultImplementationForLowCardinalityColumns() const override { return false; } bool isVariadic() const override { return true; } size_t getNumberOfArguments() const override { return 0; } DataTypePtr getReturnTypeImpl(const DataTypes & /*arguments*/) const override { return std::make_shared(); } ColumnPtr executeImpl(const ColumnsWithTypeAndName & arguments, const DataTypePtr & result_type, size_t input_rows_count) const override { size_t num_args = arguments.size(); /// If the resulting size is constant, return constant column. bool all_constant = true; UInt64 constant_size = 0; for (size_t arg_num = 0; arg_num < num_args; ++arg_num) { if (arguments[arg_num].type->isValueUnambiguouslyRepresentedInFixedSizeContiguousMemoryRegion()) { constant_size += arguments[arg_num].type->getSizeOfValueInMemory(); } else { all_constant = false; break; } } if (all_constant) return result_type->createColumnConst(input_rows_count, constant_size); auto result_col = ColumnUInt64::create(input_rows_count); auto & vec_res = result_col->getData(); for (size_t arg_num = 0; arg_num < num_args; ++arg_num) { const IColumn * column = arguments[arg_num].column.get(); if (arg_num == 0) for (size_t row_num = 0; row_num < input_rows_count; ++row_num) vec_res[row_num] = column->byteSizeAt(row_num); else for (size_t row_num = 0; row_num < input_rows_count; ++row_num) vec_res[row_num] += column->byteSizeAt(row_num); } return result_col; } }; } void registerFunctionByteSize(FunctionFactory & factory) { factory.registerFunction(); } }