com.hazelcast.org.apache.calcite.sql.fun.SqlJsonValueFunction Maven / Gradle / Ivy
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to you under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.hazelcast.org.apache.calcite.sql.fun;
import com.hazelcast.org.apache.calcite.rel.type.RelDataType;
import com.hazelcast.org.apache.calcite.rel.type.RelDataTypeFactory;
import com.hazelcast.org.apache.calcite.sql.SqlCall;
import com.hazelcast.org.apache.calcite.sql.SqlFunction;
import com.hazelcast.org.apache.calcite.sql.SqlFunctionCategory;
import com.hazelcast.org.apache.calcite.sql.SqlJsonValueReturning;
import com.hazelcast.org.apache.calcite.sql.SqlKind;
import com.hazelcast.org.apache.calcite.sql.SqlLiteral;
import com.hazelcast.org.apache.calcite.sql.SqlNode;
import com.hazelcast.org.apache.calcite.sql.SqlOperandCountRange;
import com.hazelcast.org.apache.calcite.sql.SqlOperatorBinding;
import com.hazelcast.org.apache.calcite.sql.SqlWriter;
import com.hazelcast.org.apache.calcite.sql.type.OperandTypes;
import com.hazelcast.org.apache.calcite.sql.type.ReturnTypes;
import com.hazelcast.org.apache.calcite.sql.type.SqlOperandCountRanges;
import com.hazelcast.org.apache.calcite.sql.type.SqlTypeFamily;
import com.hazelcast.org.apache.calcite.sql.type.SqlTypeName;
import com.hazelcast.org.apache.calcite.sql.type.SqlTypeTransforms;
import com.hazelcast.com.google.common.collect.ImmutableList;
import com.hazelcast.org.checkerframework.checker.nullness.qual.Nullable;
import java.util.Arrays;
import java.util.List;
import java.util.Objects;
import java.util.Optional;
import java.util.stream.Collectors;
/**
* The JSON_VALUE
function.
*/
public class SqlJsonValueFunction extends SqlFunction {
public SqlJsonValueFunction(String name) {
super(name, SqlKind.OTHER_FUNCTION,
ReturnTypes.cascade(
opBinding -> explicitTypeSpec(opBinding).orElse(getDefaultType(opBinding)),
SqlTypeTransforms.FORCE_NULLABLE),
null,
OperandTypes.family(
ImmutableList.of(SqlTypeFamily.ANY, SqlTypeFamily.CHARACTER),
ordinal -> ordinal > 1),
SqlFunctionCategory.SYSTEM);
}
/** Returns VARCHAR(2000) as default. */
private static RelDataType getDefaultType(SqlOperatorBinding opBinding) {
final RelDataTypeFactory typeFactory = opBinding.getTypeFactory();
return typeFactory.createSqlType(SqlTypeName.VARCHAR, 2000);
}
/**
* Returns new operand list with type specification removed.
*/
public static List removeTypeSpecOperands(SqlCall call) {
@Nullable SqlNode[] operands = call.getOperandList().toArray(new SqlNode[0]);
if (hasExplicitTypeSpec(operands)) {
operands[2] = null;
operands[3] = null;
}
return Arrays.stream(operands)
.filter(Objects::nonNull)
.collect(Collectors.toList());
}
@Override public SqlOperandCountRange getOperandCountRange() {
return SqlOperandCountRanges.between(2, 10);
}
/** Returns the optional explicit returning type specification. **/
private static Optional explicitTypeSpec(SqlOperatorBinding opBinding) {
if (opBinding.getOperandCount() > 2
&& opBinding.isOperandLiteral(2, false)
&& opBinding.getOperandLiteralValue(2, Object.class)
instanceof SqlJsonValueReturning) {
return Optional.of(opBinding.getOperandType(3));
}
return Optional.empty();
}
/** Returns whether there is an explicit return type specification. */
public static boolean hasExplicitTypeSpec(@Nullable SqlNode[] operands) {
return operands.length > 2
&& isReturningTypeSymbol(operands[2]);
}
private static boolean isReturningTypeSymbol(@Nullable SqlNode node) {
return node instanceof SqlLiteral
&& ((SqlLiteral) node).getValue() instanceof SqlJsonValueReturning;
}
@Override public String getAllowedSignatures(String opNameToUse) {
return "JSON_VALUE(json_doc, path [RETURNING type] "
+ "[{NULL | ERROR | DEFAULT value} ON EMPTY] "
+ "[{NULL | ERROR | DEFAULT value} ON ERROR])";
}
@Override public void unparse(SqlWriter writer, SqlCall call, int leftPrec, int rightPrec) {
final SqlWriter.Frame frame = writer.startFunCall(getName());
call.operand(0).unparse(writer, leftPrec, rightPrec);
writer.sep(",", true);
for (int i = 1; i < call.operandCount(); i++) {
call.operand(i).unparse(writer, leftPrec, rightPrec);
}
writer.endFunCall(frame);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy