risingwave_expr_impl/scalar/jsonb_record.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 risingwave_common::types::{JsonbRef, MapRef, MapValue, Scalar, StructRef, StructValue};
16use risingwave_expr::expr::Context;
17use risingwave_expr::{ExprError, Result, function};
18
19/// Expands the top-level JSON object to a row having the composite type of the base argument.
20/// The JSON object is scanned for fields whose names match column names of the output row type,
21/// and their values are inserted into those columns of the output. (Fields that do not correspond
22/// to any output column name are ignored.) In typical use, the value of base is just NULL, which
23/// means that any output columns that do not match any object field will be filled with nulls.
24/// However, if base isn't NULL then the values it contains will be used for unmatched columns.
25///
26/// # Examples
27///
28/// ```slt
29/// query ITT
30/// select (jsonb_populate_record(
31/// null::struct<a int, b text[], c struct<d int, e text>>,
32/// '{"a": 1, "b": ["2", "a b"], "c": {"d": 4, "e": "a b c"}, "x": "foo"}'
33/// )).*;
34/// ----
35/// 1 {2,"a b"} (4,"a b c")
36///
37/// query ITT
38/// select (jsonb_populate_record(
39/// row(1, null, row(4, '5'))::struct<a int, b text[], c struct<d int, e text>>,
40/// '{"b": ["2", "a b"], "c": {"e": "a b c"}, "x": "foo"}'
41/// )).*;
42/// ----
43/// 1 {2,"a b"} (4,"a b c")
44///
45/// query II
46/// select * from jsonb_populate_record(
47/// null::struct<a int, b int>,
48/// '{"a": 1, "b": 2}'
49/// );
50/// ----
51/// 1 2
52/// ```
53#[function("jsonb_populate_record(struct, jsonb) -> struct")]
54fn jsonb_populate_record(
55 base: Option<StructRef<'_>>,
56 jsonb: JsonbRef<'_>,
57 ctx: &Context,
58) -> Result<StructValue> {
59 let output_type = ctx.return_type.as_struct();
60 jsonb.populate_struct(output_type, base).map_err(parse_err)
61}
62
63#[function("jsonb_populate_map(anymap, jsonb) -> anymap")]
64pub fn jsonb_populate_map(
65 base: Option<MapRef<'_>>,
66 v: JsonbRef<'_>,
67 ctx: &Context,
68) -> Result<MapValue> {
69 let output_type = ctx.return_type.as_map();
70 let jsonb_map = v
71 .to_map(output_type)
72 .map_err(|e| ExprError::Parse(e.into()))?;
73 match base {
74 Some(base) => Ok(MapValue::concat(base, jsonb_map.as_scalar_ref())),
75 None => Ok(jsonb_map),
76 }
77}
78
79/// Expands the top-level JSON array of objects to a set of rows having the composite type of the
80/// base argument. Each element of the JSON array is processed as described above for
81/// `jsonb_populate_record`.
82///
83/// # Examples
84///
85/// ```slt
86/// query II
87/// select * from jsonb_populate_recordset(
88/// null::struct<a int, b int>,
89/// '[{"a":1,"b":2}, {"a":3,"b":4}]'::jsonb
90/// );
91/// ----
92/// 1 2
93/// 3 4
94///
95/// query II
96/// select * from jsonb_populate_recordset(
97/// row(0, 0)::struct<a int, b int>,
98/// '[{}, {"a":1}, {"b":2}, {"a":1,"b":2}]'::jsonb
99/// );
100/// ----
101/// 0 0
102/// 1 0
103/// 0 2
104/// 1 2
105/// ```
106#[function("jsonb_populate_recordset(struct, jsonb) -> setof struct")]
107fn jsonb_populate_recordset<'a>(
108 base: Option<StructRef<'a>>,
109 jsonb: JsonbRef<'a>,
110 ctx: &'a Context,
111) -> Result<impl Iterator<Item = Result<StructValue>> + 'a> {
112 let output_type = ctx.return_type.as_struct();
113 Ok(jsonb
114 .array_elements()
115 .map_err(parse_err)?
116 .map(move |elem| elem.populate_struct(output_type, base).map_err(parse_err)))
117}
118
119/// Expands the top-level JSON object to a row having the composite type defined by an AS clause.
120/// The output record is filled from fields of the JSON object, in the same way as described above
121/// for `jsonb_populate_record`. Since there is no input record value, unmatched columns are always
122/// filled with nulls.
123///
124/// # Examples
125///
126/// // FIXME(runji): this query is blocked by parser and frontend support.
127/// ```slt,ignore
128/// query T
129/// select * from jsonb_to_record('{"a":1,"b":[1,2,3],"c":[1,2,3],"e":"bar","r": {"a": 123, "b": "a b c"}}')
130/// as x(a int, b text, c int[], d text, r struct<a int, b text>);
131/// ----
132/// 1 [1,2,3] {1,2,3} NULL (123,"a b c")
133/// ```
134#[function("jsonb_to_record(jsonb) -> struct", type_infer = "unreachable")]
135fn jsonb_to_record(jsonb: JsonbRef<'_>, ctx: &Context) -> Result<StructValue> {
136 let output_type = ctx.return_type.as_struct();
137 jsonb.to_struct(output_type).map_err(parse_err)
138}
139
140/// Expands the top-level JSON array of objects to a set of rows having the composite type defined
141/// by an AS clause. Each element of the JSON array is processed as described above for
142/// `jsonb_populate_record`.
143///
144/// # Examples
145///
146/// // FIXME(runji): this query is blocked by parser and frontend support.
147/// ```slt,ignore
148/// query IT
149/// select * from jsonb_to_recordset('[{"a":1,"b":"foo"}, {"a":"2","c":"bar"}]') as x(a int, b text);
150/// ----
151/// 1 foo
152/// 2 NULL
153/// ```
154#[function(
155 "jsonb_to_recordset(jsonb) -> setof struct",
156 type_infer = "unreachable"
157)]
158fn jsonb_to_recordset<'a>(
159 jsonb: JsonbRef<'a>,
160 ctx: &'a Context,
161) -> Result<impl Iterator<Item = Result<StructValue>> + 'a> {
162 let output_type = ctx.return_type.as_struct();
163 Ok(jsonb
164 .array_elements()
165 .map_err(parse_err)?
166 .map(|elem| elem.to_struct(output_type).map_err(parse_err)))
167}
168
169/// Construct a parse error from String.
170fn parse_err(s: String) -> ExprError {
171 ExprError::Parse(s.into())
172}