Coverage Report

Created: 2026-03-15 17:28

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