Coverage Report

Created: 2026-03-14 18:33

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
be/src/exprs/function/function_struct.cpp
Line
Count
Source
1
// Licensed to the Apache Software Foundation (ASF) under one
2
// or more contributor license agreements.  See the NOTICE file
3
// distributed with this work for additional information
4
// regarding copyright ownership.  The ASF licenses this file
5
// to you under the Apache License, Version 2.0 (the
6
// "License"); you may not use this file except in compliance
7
// with the License.  You may obtain a copy of the License at
8
//
9
//   http://www.apache.org/licenses/LICENSE-2.0
10
//
11
// Unless required by applicable law or agreed to in writing,
12
// software distributed under the License is distributed on an
13
// "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
14
// KIND, either express or implied.  See the License for the
15
// specific language governing permissions and limitations
16
// under the License.
17
18
#include <glog/logging.h>
19
#include <stddef.h>
20
21
#include <memory>
22
#include <ostream>
23
#include <string>
24
#include <utility>
25
26
#include "common/status.h"
27
#include "core/block/block.h"
28
#include "core/block/column_numbers.h"
29
#include "core/block/column_with_type_and_name.h"
30
#include "core/column/column.h"
31
#include "core/column/column_nullable.h"
32
#include "core/column/column_struct.h"
33
#include "core/column/column_vector.h"
34
#include "core/data_type/data_type.h"
35
#include "core/data_type/data_type_nullable.h"
36
#include "core/data_type/data_type_struct.h"
37
#include "core/typeid_cast.h"
38
#include "core/types.h"
39
#include "exprs/aggregate/aggregate_function.h"
40
#include "exprs/function/function.h"
41
#include "exprs/function/simple_function_factory.h"
42
43
namespace doris {
44
class FunctionContext;
45
} // namespace doris
46
47
namespace doris {
48
49
// construct a struct
50
template <typename Impl>
51
class FunctionStruct : public IFunction {
52
public:
53
    static constexpr auto name = Impl::name;
54
1.40k
    static FunctionPtr create() { return std::make_shared<FunctionStruct>(); }
_ZN5doris14FunctionStructINS_10StructImplEE6createEv
Line
Count
Source
54
666
    static FunctionPtr create() { return std::make_shared<FunctionStruct>(); }
_ZN5doris14FunctionStructINS_15NamedStructImplEE6createEv
Line
Count
Source
54
740
    static FunctionPtr create() { return std::make_shared<FunctionStruct>(); }
55
56
    /// Get function name.
57
0
    String get_name() const override { return name; }
Unexecuted instantiation: _ZNK5doris14FunctionStructINS_10StructImplEE8get_nameB5cxx11Ev
Unexecuted instantiation: _ZNK5doris14FunctionStructINS_15NamedStructImplEE8get_nameB5cxx11Ev
58
59
2
    bool is_variadic() const override { return true; }
_ZNK5doris14FunctionStructINS_10StructImplEE11is_variadicEv
Line
Count
Source
59
1
    bool is_variadic() const override { return true; }
_ZNK5doris14FunctionStructINS_15NamedStructImplEE11is_variadicEv
Line
Count
Source
59
1
    bool is_variadic() const override { return true; }
60
61
2.78k
    bool use_default_implementation_for_nulls() const override { return false; }
_ZNK5doris14FunctionStructINS_10StructImplEE36use_default_implementation_for_nullsEv
Line
Count
Source
61
1.33k
    bool use_default_implementation_for_nulls() const override { return false; }
_ZNK5doris14FunctionStructINS_15NamedStructImplEE36use_default_implementation_for_nullsEv
Line
Count
Source
61
1.45k
    bool use_default_implementation_for_nulls() const override { return false; }
62
63
0
    size_t get_number_of_arguments() const override { return 0; }
Unexecuted instantiation: _ZNK5doris14FunctionStructINS_10StructImplEE23get_number_of_argumentsEv
Unexecuted instantiation: _ZNK5doris14FunctionStructINS_15NamedStructImplEE23get_number_of_argumentsEv
64
65
1.38k
    void check_number_of_arguments(size_t number_of_arguments) const override {
66
1.38k
        DCHECK(number_of_arguments > 0)
67
0
                << "function: " << get_name() << ", arguments should not be empty.";
68
1.38k
        return Impl::check_number_of_arguments(number_of_arguments);
69
1.38k
    }
_ZNK5doris14FunctionStructINS_10StructImplEE25check_number_of_argumentsEm
Line
Count
Source
65
657
    void check_number_of_arguments(size_t number_of_arguments) const override {
66
657
        DCHECK(number_of_arguments > 0)
67
0
                << "function: " << get_name() << ", arguments should not be empty.";
68
657
        return Impl::check_number_of_arguments(number_of_arguments);
69
657
    }
_ZNK5doris14FunctionStructINS_15NamedStructImplEE25check_number_of_argumentsEm
Line
Count
Source
65
731
    void check_number_of_arguments(size_t number_of_arguments) const override {
66
731
        DCHECK(number_of_arguments > 0)
67
0
                << "function: " << get_name() << ", arguments should not be empty.";
68
731
        return Impl::check_number_of_arguments(number_of_arguments);
69
731
    }
70
71
1.38k
    DataTypePtr get_return_type_impl(const DataTypes& arguments) const override {
72
1.38k
        return Impl::get_return_type_impl(arguments);
73
1.38k
    }
_ZNK5doris14FunctionStructINS_10StructImplEE20get_return_type_implERKSt6vectorISt10shared_ptrIKNS_9IDataTypeEESaIS7_EE
Line
Count
Source
71
657
    DataTypePtr get_return_type_impl(const DataTypes& arguments) const override {
72
657
        return Impl::get_return_type_impl(arguments);
73
657
    }
_ZNK5doris14FunctionStructINS_15NamedStructImplEE20get_return_type_implERKSt6vectorISt10shared_ptrIKNS_9IDataTypeEESaIS7_EE
Line
Count
Source
71
731
    DataTypePtr get_return_type_impl(const DataTypes& arguments) const override {
72
731
        return Impl::get_return_type_impl(arguments);
73
731
    }
74
75
    Status execute_impl(FunctionContext* context, Block& block, const ColumnNumbers& arguments,
76
1.39k
                        uint32_t result, size_t input_rows_count) const override {
77
1.39k
        auto result_col = block.get_by_position(result).type->create_column();
78
1.39k
        auto struct_column = assert_cast<ColumnStruct*>(result_col.get());
79
1.39k
        ColumnNumbers args_num;
80
10.2k
        for (size_t i = 0; i < arguments.size(); i++) {
81
8.87k
            if (Impl::pred(i)) {
82
6.77k
                args_num.push_back(arguments[i]);
83
6.77k
            }
84
8.87k
        }
85
1.39k
        size_t num_element = args_num.size();
86
1.39k
        if (num_element != struct_column->tuple_size()) {
87
0
            return Status::RuntimeError(
88
0
                    "function {} args number {} is not equal to result struct field number {}.",
89
0
                    get_name(), num_element, struct_column->tuple_size());
90
0
        }
91
1.39k
        std::vector<ColumnPtr> arg(num_element);
92
8.17k
        for (size_t i = 0; i < num_element; ++i) {
93
6.77k
            auto& nested_col = struct_column->get_column(i);
94
6.77k
            nested_col.reserve(input_rows_count);
95
6.77k
            bool is_nullable = nested_col.is_nullable();
96
6.77k
            auto& col = block.get_by_position(args_num[i]).column;
97
6.77k
            col = col->convert_to_full_column_if_const();
98
6.77k
            arg[i] = col;
99
6.77k
            if (is_nullable && !col->is_nullable()) {
100
5.57k
                arg[i] = ColumnNullable::create(col, ColumnUInt8::create(col->size(), 0));
101
5.57k
            }
102
6.77k
        }
103
104
        // insert value into struct column by column
105
8.17k
        for (size_t i = 0; i < num_element; ++i) {
106
6.77k
            struct_column->get_column(i).insert_range_from(*arg[i], 0, input_rows_count);
107
6.77k
        }
108
1.39k
        block.replace_by_position(result, std::move(result_col));
109
1.39k
        return Status::OK();
110
1.39k
    }
_ZNK5doris14FunctionStructINS_10StructImplEE12execute_implEPNS_15FunctionContextERNS_5BlockERKSt6vectorIjSaIjEEjm
Line
Count
Source
76
675
                        uint32_t result, size_t input_rows_count) const override {
77
675
        auto result_col = block.get_by_position(result).type->create_column();
78
675
        auto struct_column = assert_cast<ColumnStruct*>(result_col.get());
79
675
        ColumnNumbers args_num;
80
5.34k
        for (size_t i = 0; i < arguments.size(); i++) {
81
4.66k
            if (Impl::pred(i)) {
82
4.66k
                args_num.push_back(arguments[i]);
83
4.66k
            }
84
4.66k
        }
85
675
        size_t num_element = args_num.size();
86
675
        if (num_element != struct_column->tuple_size()) {
87
0
            return Status::RuntimeError(
88
0
                    "function {} args number {} is not equal to result struct field number {}.",
89
0
                    get_name(), num_element, struct_column->tuple_size());
90
0
        }
91
675
        std::vector<ColumnPtr> arg(num_element);
92
5.34k
        for (size_t i = 0; i < num_element; ++i) {
93
4.66k
            auto& nested_col = struct_column->get_column(i);
94
4.66k
            nested_col.reserve(input_rows_count);
95
4.66k
            bool is_nullable = nested_col.is_nullable();
96
4.66k
            auto& col = block.get_by_position(args_num[i]).column;
97
4.66k
            col = col->convert_to_full_column_if_const();
98
4.66k
            arg[i] = col;
99
4.66k
            if (is_nullable && !col->is_nullable()) {
100
3.57k
                arg[i] = ColumnNullable::create(col, ColumnUInt8::create(col->size(), 0));
101
3.57k
            }
102
4.66k
        }
103
104
        // insert value into struct column by column
105
5.34k
        for (size_t i = 0; i < num_element; ++i) {
106
4.66k
            struct_column->get_column(i).insert_range_from(*arg[i], 0, input_rows_count);
107
4.66k
        }
108
675
        block.replace_by_position(result, std::move(result_col));
109
675
        return Status::OK();
110
675
    }
_ZNK5doris14FunctionStructINS_15NamedStructImplEE12execute_implEPNS_15FunctionContextERNS_5BlockERKSt6vectorIjSaIjEEjm
Line
Count
Source
76
723
                        uint32_t result, size_t input_rows_count) const override {
77
723
        auto result_col = block.get_by_position(result).type->create_column();
78
723
        auto struct_column = assert_cast<ColumnStruct*>(result_col.get());
79
723
        ColumnNumbers args_num;
80
4.93k
        for (size_t i = 0; i < arguments.size(); i++) {
81
4.21k
            if (Impl::pred(i)) {
82
2.10k
                args_num.push_back(arguments[i]);
83
2.10k
            }
84
4.21k
        }
85
723
        size_t num_element = args_num.size();
86
723
        if (num_element != struct_column->tuple_size()) {
87
0
            return Status::RuntimeError(
88
0
                    "function {} args number {} is not equal to result struct field number {}.",
89
0
                    get_name(), num_element, struct_column->tuple_size());
90
0
        }
91
723
        std::vector<ColumnPtr> arg(num_element);
92
2.82k
        for (size_t i = 0; i < num_element; ++i) {
93
2.10k
            auto& nested_col = struct_column->get_column(i);
94
2.10k
            nested_col.reserve(input_rows_count);
95
2.10k
            bool is_nullable = nested_col.is_nullable();
96
2.10k
            auto& col = block.get_by_position(args_num[i]).column;
97
2.10k
            col = col->convert_to_full_column_if_const();
98
2.10k
            arg[i] = col;
99
2.10k
            if (is_nullable && !col->is_nullable()) {
100
2.00k
                arg[i] = ColumnNullable::create(col, ColumnUInt8::create(col->size(), 0));
101
2.00k
            }
102
2.10k
        }
103
104
        // insert value into struct column by column
105
2.82k
        for (size_t i = 0; i < num_element; ++i) {
106
2.10k
            struct_column->get_column(i).insert_range_from(*arg[i], 0, input_rows_count);
107
2.10k
        }
108
723
        block.replace_by_position(result, std::move(result_col));
109
723
        return Status::OK();
110
723
    }
111
};
112
113
// struct(value1, value2, value3) -> {value1, value2, value3}
114
struct StructImpl {
115
    static constexpr auto name = "struct";
116
4.66k
    static constexpr auto pred = [](size_t i) { return true; };
117
118
657
    static void check_number_of_arguments(size_t number_of_arguments) {}
119
120
657
    static DataTypePtr get_return_type_impl(const DataTypes& arguments) {
121
657
        return std::make_shared<DataTypeStruct>(make_nullable(arguments));
122
657
    }
123
};
124
125
// named_struct(name1, value1, name2, value2) -> {name1:value1, name2:value2}
126
struct NamedStructImpl {
127
    static constexpr auto name = "named_struct";
128
4.21k
    static constexpr auto pred = [](size_t i) { return (i & 1) == 1; };
129
130
731
    static void check_number_of_arguments(size_t number_of_arguments) {
131
731
        DCHECK(number_of_arguments % 2 == 0)
132
0
                << "function: " << name << ", arguments size should be even number.";
133
731
    }
134
135
731
    static DataTypePtr get_return_type_impl(const DataTypes& arguments) {
136
731
        DataTypes data_types(arguments.size() / 2);
137
731
        size_t even_idx = 1;
138
2.84k
        for (size_t i = 0; i < data_types.size(); i++) {
139
2.11k
            data_types[i] = arguments[even_idx];
140
2.11k
            even_idx += 2;
141
2.11k
        }
142
731
        return std::make_shared<DataTypeStruct>(make_nullable(data_types));
143
731
    }
144
};
145
146
8
void register_function_struct(SimpleFunctionFactory& factory) {
147
8
    factory.register_function<FunctionStruct<StructImpl>>();
148
8
    factory.register_function<FunctionStruct<NamedStructImpl>>();
149
8
}
150
151
} // namespace doris