All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.google.zetasql.Function Maven / Gradle / Ivy

There is a newer version: 2024.11.1
Show newest version
/*
 * Copyright 2019 ZetaSQL Authors
 *
 * Licensed 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.google.zetasql;

import com.google.common.base.Joiner;
import com.google.common.base.Preconditions;
import com.google.common.collect.ImmutableList;
import com.google.zetasql.FunctionProtos.FunctionOptionsProto;
import com.google.zetasql.FunctionProtos.FunctionProto;
import com.google.zetasql.FunctionProtos.FunctionSignatureProto;
import com.google.zetasql.ZetaSQLFunctions.FunctionEnums.Mode;
import com.google.zetasql.ZetaSQLFunctions.FunctionEnums.WindowOrderSupport;
import java.io.Serializable;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;

/**
 * The Function interface identifies the functions available in a query engine. Each Function
 * includes a set of FunctionSignatures, where a signature indicates:
 *
 * 
    *
  1. Argument and result types. *
  2. A 'context' for the signature. *
* *

A Function also indicates the 'group' it belongs to. * *

The Function also includes {@code } for specifying additional resolution * requirements, if any. Additionally, {@code } can identify a function alias/synonym that * Catalogs must expose for function lookups by name. */ public class Function implements Serializable { static final String ZETASQL_FUNCTION_GROUP_NAME = "ZetaSQL"; private final ImmutableList namePath; private final String group; private final Mode mode; private final ImmutableList signatures; private final FunctionOptionsProto options; /** * Construct a Function. * @param namePath Name (and namespaces) of the function. * @param group * @param mode Enum value, one of SCALAR, AGGREGATE, ANALYTIC. * @param signatures * @param options */ public Function( List namePath, String group, Mode mode, List signatures, FunctionOptionsProto options) { Preconditions.checkArgument(!namePath.isEmpty()); this.namePath = ImmutableList.copyOf(namePath); this.group = Preconditions.checkNotNull(group); this.mode = Preconditions.checkNotNull(mode); this.signatures = ImmutableList.copyOf(signatures); this.options = Preconditions.checkNotNull(options); checkWindowSupportOptions(); } public FunctionProto serialize(FileDescriptorSetsBuilder fileDescriptorSetsBuilder) { FunctionProto.Builder builder = FunctionProto.newBuilder() .addAllNamePath(namePath) .setGroup(group) .setMode(mode) .setOptions(options); for (FunctionSignature signature : signatures) { builder.addSignature(signature.serialize(fileDescriptorSetsBuilder)); } if (group.equals(TemplatedSQLFunction.TEMPLATED_SQL_FUNCTION_GROUP)) { builder.setParseResumeLocation(((TemplatedSQLFunction) this).parseResumeLocation.serialize()); for (String name : ((TemplatedSQLFunction) this).argumentNames) { builder.addTemplatedSqlFunctionArgumentName(name); } } return builder.build(); } public static Function deserialize( FunctionProto proto, final ImmutableList pools) { if (proto.getGroup().equals(TemplatedSQLFunction.TEMPLATED_SQL_FUNCTION_GROUP)) { return TemplatedSQLFunction.deserialize(proto, pools); } List signatures = new ArrayList<>(); for (FunctionSignatureProto signature : proto.getSignatureList()) { signatures.add(FunctionSignature.deserialize(signature, pools)); } return new Function( proto.getNamePathList(), proto.getGroup(), proto.getMode(), signatures, proto.getOptions()); } public Function( List namePath, String group, Mode mode, List signatures) { this(namePath, group, mode, signatures, FunctionOptionsProto.getDefaultInstance()); } public Function( String name, String group, Mode mode, List signatures, FunctionOptionsProto options) { this(Arrays.asList(name), group, mode, signatures, options); } public Function( String name, String group, Mode mode, List signatures) { this(Arrays.asList(name), group, mode, signatures); } public Function(String name, String group, Mode mode) { this(name, group, mode, ImmutableList.of()); } public String getName() { return namePath.get(namePath.size() - 1); } public ImmutableList getNamePath() { return namePath; } public String getFullName() { return getFullName(true); } public String getFullName(boolean includeGroup) { String pathname = Joiner.on('.').join(namePath); if (includeGroup) { return group + ":" + pathname; } else { return pathname; } } public String getSqlName() { String name; if (!options.getSqlName().isEmpty()) { name = options.getSqlName(); } else if (getName().startsWith("$")) { // The name starts with '$' so it is an internal function name. Strip off // the leading '$' and convert all '_' to ' '. name = getName().substring(1).replace('_', ' '); } else if (isZetaSQLBuiltin()) { name = getFullName(/* includeGroup */ false); } else { name = getFullName(); } if (options.getUsesUpperCaseSqlName()) { name = name.toUpperCase(); } return name; } public String getGroup() { return group; } public ImmutableList getSignatureList() { return signatures; } public Mode getMode() { return mode; } public FunctionOptionsProto getOptions() { return options; } public boolean isScalar() { return mode == Mode.SCALAR; } public boolean isAggregate() { return mode == Mode.AGGREGATE; } public boolean isAnalytic() { return mode == Mode.ANALYTIC; } public String debugString(boolean verbose) { if (!verbose) { return getFullName(); } StringBuilder builder = new StringBuilder(getFullName()); String alias = options.getAliasName(); if (!alias.isEmpty()) { builder.append('|').append(alias); } if (!signatures.isEmpty()) { builder.append('\n').append(FunctionSignature.signaturesToString(signatures)); } return builder.toString(); } @Override public String toString() { return debugString(false); } public boolean isZetaSQLBuiltin() { return group.equals(ZETASQL_FUNCTION_GROUP_NAME); } public boolean supportsWindowOrdering() { WindowOrderSupport windowOrderingSupport = options.getWindowOrderingSupport(); return windowOrderingSupport == WindowOrderSupport.ORDER_REQUIRED || windowOrderingSupport == WindowOrderSupport.ORDER_OPTIONAL; } public boolean requireWindowOrdering() { return options.getWindowOrderingSupport() == WindowOrderSupport.ORDER_REQUIRED; } private void checkWindowSupportOptions() { if (isScalar() && options.getSupportsOverClause()) { throw new IllegalArgumentException("Scalar functions cannot support over clause"); } if (isAnalytic() && !options.getSupportsOverClause()) { throw new IllegalArgumentException("Analytic functions must support over clause"); } } /** * This represents a templated function with a SQL body. * *

The purpose of this class is to help support statements of the form "CREATE FUNCTION * () AS ", where the may have templated types like "ANY * TYPE". In this case, ZetaSQL cannot resolve the function expression right away and must defer * this work until later when the function is called with concrete argument types. */ public static class TemplatedSQLFunction extends Function { private final ImmutableList argumentNames; private final ParseResumeLocation parseResumeLocation; public static final String TEMPLATED_SQL_FUNCTION_GROUP = "Templated_SQL_Function"; public TemplatedSQLFunction( ImmutableList namePath, FunctionSignature signature, ImmutableList argumentNames, ParseResumeLocation parseResumeLocation) { super(namePath, TEMPLATED_SQL_FUNCTION_GROUP, Mode.SCALAR, ImmutableList.of(signature)); this.argumentNames = argumentNames; this.parseResumeLocation = parseResumeLocation; } public ImmutableList getArgumentNames() { return argumentNames; } public String getSqlBody() { return this.parseResumeLocation.getInput().substring(this.parseResumeLocation.getBytePosition()); } /* Deserializes this function from a protocol buffer. */ public static TemplatedSQLFunction deserialize( FunctionProto proto, final ImmutableList pools) { Preconditions.checkArgument(proto.getGroup().equals(TEMPLATED_SQL_FUNCTION_GROUP), proto); ImmutableList namePath = ImmutableList.copyOf(proto.getNamePathList()); FunctionSignature signature = FunctionSignature.deserialize(proto.getSignature(0), pools); ImmutableList.Builder builder = new ImmutableList.Builder(); for (String name : proto.getTemplatedSqlFunctionArgumentNameList()) { builder.add(name); } return new TemplatedSQLFunction( namePath, signature, builder.build(), new ParseResumeLocation(proto.getParseResumeLocation())); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy