Coverage Report

Created: 2026-03-12 15:22

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
2.44k
    static FunctionPtr create() { return std::make_shared<FunctionStruct>(); }
_ZN5doris14FunctionStructINS_10StructImplEE6createEv
Line
Count
Source
54
713
    static FunctionPtr create() { return std::make_shared<FunctionStruct>(); }
_ZN5doris14FunctionStructINS_15NamedStructImplEE6createEv
Line
Count
Source
54
1.73k
    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
4.88k
    bool use_default_implementation_for_nulls() const override { return false; }
_ZNK5doris14FunctionStructINS_10StructImplEE36use_default_implementation_for_nullsEv
Line
Count
Source
61
1.43k
    bool use_default_implementation_for_nulls() const override { return false; }
_ZNK5doris14FunctionStructINS_15NamedStructImplEE36use_default_implementation_for_nullsEv
Line
Count
Source
61
3.44k
    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
2.43k
    void check_number_of_arguments(size_t number_of_arguments) const override {
66
2.43k
        DCHECK(number_of_arguments > 0)
67
0
                << "function: " << get_name() << ", arguments should not be empty.";
68
2.43k
        return Impl::check_number_of_arguments(number_of_arguments);
69
2.43k
    }
_ZNK5doris14FunctionStructINS_10StructImplEE25check_number_of_argumentsEm
Line
Count
Source
65
705
    void check_number_of_arguments(size_t number_of_arguments) const override {
66
705
        DCHECK(number_of_arguments > 0)
67
0
                << "function: " << get_name() << ", arguments should not be empty.";
68
705
        return Impl::check_number_of_arguments(number_of_arguments);
69
705
    }
_ZNK5doris14FunctionStructINS_15NamedStructImplEE25check_number_of_argumentsEm
Line
Count
Source
65
1.72k
    void check_number_of_arguments(size_t number_of_arguments) const override {
66
1.72k
        DCHECK(number_of_arguments > 0)
67
0
                << "function: " << get_name() << ", arguments should not be empty.";
68
1.72k
        return Impl::check_number_of_arguments(number_of_arguments);
69
1.72k
    }
70
71
2.43k
    DataTypePtr get_return_type_impl(const DataTypes& arguments) const override {
72
2.43k
        return Impl::get_return_type_impl(arguments);
73
2.43k
    }
_ZNK5doris14FunctionStructINS_10StructImplEE20get_return_type_implERKSt6vectorISt10shared_ptrIKNS_9IDataTypeEESaIS7_EE
Line
Count
Source
71
705
    DataTypePtr get_return_type_impl(const DataTypes& arguments) const override {
72
705
        return Impl::get_return_type_impl(arguments);
73
705
    }
_ZNK5doris14FunctionStructINS_15NamedStructImplEE20get_return_type_implERKSt6vectorISt10shared_ptrIKNS_9IDataTypeEESaIS7_EE
Line
Count
Source
71
1.72k
    DataTypePtr get_return_type_impl(const DataTypes& arguments) const override {
72
1.72k
        return Impl::get_return_type_impl(arguments);
73
1.72k
    }
74
75
    Status execute_impl(FunctionContext* context, Block& block, const ColumnNumbers& arguments,
76
2.44k
                        uint32_t result, size_t input_rows_count) const override {
77
2.44k
        auto result_col = block.get_by_position(result).type->create_column();
78
2.44k
        auto struct_column = assert_cast<ColumnStruct*>(result_col.get());
79
2.44k
        ColumnNumbers args_num;
80
14.8k
        for (size_t i = 0; i < arguments.size(); i++) {
81
12.4k
            if (Impl::pred(i)) {
82
8.62k
                args_num.push_back(arguments[i]);
83
8.62k
            }
84
12.4k
        }
85
2.44k
        size_t num_element = args_num.size();
86
2.44k
        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
2.44k
        std::vector<ColumnPtr> arg(num_element);
92
11.0k
        for (size_t i = 0; i < num_element; ++i) {
93
8.62k
            auto& nested_col = struct_column->get_column(i);
94
8.62k
            nested_col.reserve(input_rows_count);
95
8.62k
            bool is_nullable = nested_col.is_nullable();
96
8.62k
            auto& col = block.get_by_position(args_num[i]).column;
97
8.62k
            col = col->convert_to_full_column_if_const();
98
8.62k
            arg[i] = col;
99
8.62k
            if (is_nullable && !col->is_nullable()) {
100
7.11k
                arg[i] = ColumnNullable::create(col, ColumnUInt8::create(col->size(), 0));
101
7.11k
            }
102
8.62k
        }
103
104
        // insert value into struct column by column
105
11.0k
        for (size_t i = 0; i < num_element; ++i) {
106
8.62k
            struct_column->get_column(i).insert_range_from(*arg[i], 0, input_rows_count);
107
8.62k
        }
108
2.44k
        block.replace_by_position(result, std::move(result_col));
109
2.44k
        return Status::OK();
110
2.44k
    }
_ZNK5doris14FunctionStructINS_10StructImplEE12execute_implEPNS_15FunctionContextERNS_5BlockERKSt6vectorIjSaIjEEjm
Line
Count
Source
76
728
                        uint32_t result, size_t input_rows_count) const override {
77
728
        auto result_col = block.get_by_position(result).type->create_column();
78
728
        auto struct_column = assert_cast<ColumnStruct*>(result_col.get());
79
728
        ColumnNumbers args_num;
80
5.53k
        for (size_t i = 0; i < arguments.size(); i++) {
81
4.80k
            if (Impl::pred(i)) {
82
4.80k
                args_num.push_back(arguments[i]);
83
4.80k
            }
84
4.80k
        }
85
728
        size_t num_element = args_num.size();
86
728
        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
728
        std::vector<ColumnPtr> arg(num_element);
92
5.53k
        for (size_t i = 0; i < num_element; ++i) {
93
4.80k
            auto& nested_col = struct_column->get_column(i);
94
4.80k
            nested_col.reserve(input_rows_count);
95
4.80k
            bool is_nullable = nested_col.is_nullable();
96
4.80k
            auto& col = block.get_by_position(args_num[i]).column;
97
4.80k
            col = col->convert_to_full_column_if_const();
98
4.80k
            arg[i] = col;
99
4.80k
            if (is_nullable && !col->is_nullable()) {
100
3.68k
                arg[i] = ColumnNullable::create(col, ColumnUInt8::create(col->size(), 0));
101
3.68k
            }
102
4.80k
        }
103
104
        // insert value into struct column by column
105
5.53k
        for (size_t i = 0; i < num_element; ++i) {
106
4.80k
            struct_column->get_column(i).insert_range_from(*arg[i], 0, input_rows_count);
107
4.80k
        }
108
728
        block.replace_by_position(result, std::move(result_col));
109
728
        return Status::OK();
110
728
    }
_ZNK5doris14FunctionStructINS_15NamedStructImplEE12execute_implEPNS_15FunctionContextERNS_5BlockERKSt6vectorIjSaIjEEjm
Line
Count
Source
76
1.72k
                        uint32_t result, size_t input_rows_count) const override {
77
1.72k
        auto result_col = block.get_by_position(result).type->create_column();
78
1.72k
        auto struct_column = assert_cast<ColumnStruct*>(result_col.get());
79
1.72k
        ColumnNumbers args_num;
80
9.35k
        for (size_t i = 0; i < arguments.size(); i++) {
81
7.63k
            if (Impl::pred(i)) {
82
3.81k
                args_num.push_back(arguments[i]);
83
3.81k
            }
84
7.63k
        }
85
1.72k
        size_t num_element = args_num.size();
86
1.72k
        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.72k
        std::vector<ColumnPtr> arg(num_element);
92
5.53k
        for (size_t i = 0; i < num_element; ++i) {
93
3.81k
            auto& nested_col = struct_column->get_column(i);
94
3.81k
            nested_col.reserve(input_rows_count);
95
3.81k
            bool is_nullable = nested_col.is_nullable();
96
3.81k
            auto& col = block.get_by_position(args_num[i]).column;
97
3.81k
            col = col->convert_to_full_column_if_const();
98
3.81k
            arg[i] = col;
99
3.81k
            if (is_nullable && !col->is_nullable()) {
100
3.43k
                arg[i] = ColumnNullable::create(col, ColumnUInt8::create(col->size(), 0));
101
3.43k
            }
102
3.81k
        }
103
104
        // insert value into struct column by column
105
5.53k
        for (size_t i = 0; i < num_element; ++i) {
106
3.81k
            struct_column->get_column(i).insert_range_from(*arg[i], 0, input_rows_count);
107
3.81k
        }
108
1.72k
        block.replace_by_position(result, std::move(result_col));
109
1.72k
        return Status::OK();
110
1.72k
    }
111
};
112
113
// struct(value1, value2, value3) -> {value1, value2, value3}
114
struct StructImpl {
115
    static constexpr auto name = "struct";
116
4.80k
    static constexpr auto pred = [](size_t i) { return true; };
117
118
705
    static void check_number_of_arguments(size_t number_of_arguments) {}
119
120
705
    static DataTypePtr get_return_type_impl(const DataTypes& arguments) {
121
705
        return std::make_shared<DataTypeStruct>(make_nullable(arguments));
122
705
    }
123
};
124
125
// named_struct(name1, value1, name2, value2) -> {name1:value1, name2:value2}
126
struct NamedStructImpl {
127
    static constexpr auto name = "named_struct";
128
7.63k
    static constexpr auto pred = [](size_t i) { return (i & 1) == 1; };
129
130
1.72k
    static void check_number_of_arguments(size_t number_of_arguments) {
131
1.72k
        DCHECK(number_of_arguments % 2 == 0)
132
0
                << "function: " << name << ", arguments size should be even number.";
133
1.72k
    }
134
135
1.72k
    static DataTypePtr get_return_type_impl(const DataTypes& arguments) {
136
1.72k
        DataTypes data_types(arguments.size() / 2);
137
1.72k
        size_t even_idx = 1;
138
5.55k
        for (size_t i = 0; i < data_types.size(); i++) {
139
3.82k
            data_types[i] = arguments[even_idx];
140
3.82k
            even_idx += 2;
141
3.82k
        }
142
1.72k
        return std::make_shared<DataTypeStruct>(make_nullable(data_types));
143
1.72k
    }
144
};
145
146
7
void register_function_struct(SimpleFunctionFactory& factory) {
147
7
    factory.register_function<FunctionStruct<StructImpl>>();
148
7
    factory.register_function<FunctionStruct<NamedStructImpl>>();
149
7
}
150
151
} // namespace doris