risingwave_expr_impl/table_function/
generate_subscripts.rs

1// Copyright 2025 RisingWave Labs
2//
3// Licensed under the Apache License, Version 2.0 (the "License");
4// you may not use this file except in compliance with the License.
5// You may obtain a copy of the License at
6//
7//     http://www.apache.org/licenses/LICENSE-2.0
8//
9// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS,
11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12// See the License for the specific language governing permissions and
13// limitations under the License.
14
15use auto_enums::auto_enum;
16use risingwave_common::array::ListRef;
17use risingwave_common::types::ScalarRefImpl;
18use risingwave_expr::function;
19
20/// ```slt
21/// query I
22/// SELECT generate_subscripts(ARRAY['foo', 'bar', null], -1, false) AS s;
23/// ----
24///
25/// query I
26/// SELECT generate_subscripts(ARRAY['foo', 'bar', null], 0, false) AS s;
27/// ----
28///
29/// query I
30/// SELECT generate_subscripts(ARRAY['foo', 'bar', null], 1, true) AS s;
31/// ----
32/// 3
33/// 2
34/// 1
35///
36/// query I
37/// SELECT generate_subscripts(ARRAY['foo', 'bar', null], 1, false) AS s;
38/// ----
39/// 1
40/// 2
41/// 3
42///
43/// query I
44/// SELECT generate_subscripts(ARRAY[ARRAY['foo'], ARRAY['bar'], ARRAY[null]], 0, false) AS s;
45/// ----
46///
47/// query I
48/// SELECT generate_subscripts(ARRAY[ARRAY['foo'], ARRAY['bar'], ARRAY[null]], 1, false) AS s;
49/// ----
50/// 1
51/// 2
52/// 3
53///
54/// query I
55/// SELECT generate_subscripts(ARRAY[ARRAY['foo'], ARRAY['bar'], ARRAY[null]], 2, false) AS s;
56/// ----
57/// 1
58/// ```
59#[function("generate_subscripts(anyarray, int4, boolean) -> setof int4")]
60fn generate_subscripts_reverse(
61    array: ListRef<'_>,
62    dim: i32,
63    reverse: bool,
64) -> impl Iterator<Item = i32> {
65    generate_subscripts_iterator(array, dim, reverse)
66}
67
68/// ```slt
69/// query I
70/// SELECT generate_subscripts(ARRAY['foo', 'bar', null], -1, false) AS s;
71/// ----
72///
73/// query I
74/// SELECT generate_subscripts(ARRAY['foo', 'bar', null], 0, false) AS s;
75/// ----
76///
77/// query I
78/// SELECT generate_subscripts(ARRAY['foo', 'bar', null], 1) AS s;
79/// ----
80/// 1
81/// 2
82/// 3
83///
84/// query I
85/// SELECT generate_subscripts(ARRAY['foo', 'bar', null], 1) AS s;
86/// ----
87/// 1
88/// 2
89/// 3
90///
91/// query I
92/// SELECT generate_subscripts(ARRAY[ARRAY['foo'], ARRAY['bar'], ARRAY[null]], 0) AS s;
93/// ----
94///
95/// query I
96/// SELECT generate_subscripts(ARRAY[ARRAY['foo'], ARRAY['bar'], ARRAY[null]], 1) AS s;
97/// ----
98/// 1
99/// 2
100/// 3
101///
102/// query I
103/// SELECT generate_subscripts(ARRAY[ARRAY['foo'], ARRAY['bar'], ARRAY[null]], 2) AS s;
104/// ----
105/// 1
106/// ```
107#[function("generate_subscripts(anyarray, int4) -> setof int4")]
108fn generate_subscripts(array: ListRef<'_>, dim: i32) -> impl Iterator<Item = i32> {
109    generate_subscripts_iterator(array, dim, false)
110}
111
112#[auto_enum(Iterator)]
113fn generate_subscripts_iterator(
114    array: ListRef<'_>,
115    dim: i32,
116    reverse: bool,
117) -> impl Iterator<Item = i32> {
118    let (cur, end) = generate_subscripts_inner(array, dim);
119
120    if reverse {
121        return (cur..end).rev();
122    } else {
123        return cur..end;
124    }
125}
126
127fn generate_subscripts_inner(array: ListRef<'_>, dim: i32) -> (i32, i32) {
128    let nothing = (0, 0);
129    match dim {
130        ..=0 => nothing,
131        1 => (1, array.len() as i32 + 1),
132        // Although RW's array can be zig-zag, we just look at the first element.
133        2.. => match array.get(0) {
134            Some(Some(ScalarRefImpl::List(list))) => generate_subscripts_inner(list, dim - 1),
135            _ => nothing,
136        },
137    }
138}