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

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

/*
 * 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.Preconditions;
import com.google.common.collect.HashBasedTable;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.Table;
import com.google.zetasql.ZetaSQLOptionsProto.AllowedHintsAndOptionsProto;
import com.google.zetasql.ZetaSQLOptionsProto.AllowedHintsAndOptionsProto.HintProto;
import com.google.zetasql.ZetaSQLOptionsProto.AllowedHintsAndOptionsProto.OptionProto;
import com.google.zetasql.ZetaSQLType.TypeProto;
import java.io.Serializable;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
import javax.annotation.Nullable;

/**
 * This class specifies a set of allowed hints and options, and their expected types.
 *
 * 

Each hint or option has an expected Type, which can be NULL. If the expected type is NULL, * then any type is allowed. If a type is specified, the resolved value for the hint will always * have the expected type, and the analyzer will give an error if coercion is not possible. * *

Hint, option and qualifier names are all case insensitive. The resolved AST will contain the * original case as written by the user. * *

The {@code disallow_unknown_options} and {@code disallow_unknown_hints_with_qualifiers} fields * can be set to indicate that errors should be given on unknown options or hints (with specific * qualifiers). Unknown hints with other qualifiers do not cause errors. */ public class AllowedHintsAndOptions implements Serializable { // If true, give an error for an unknown option. boolean disallowUnknownOptions = false; // Keys are lower-case qualifiers, values are the original qualifiers. private final Map disallowUnknownHintsWithQualifiers = new HashMap<>(); // Table containing declared hints. Table keys are in lower-case. The key is // (qualifier, hint). Unqualified hints are declared using an empty qualifier. private final Table hints = HashBasedTable.create(); // Map containing declared options. Map keys are in lower-case. private final Map options = new HashMap<>(); public AllowedHintsAndOptions() {} /** * This is recommended constructor to use for normal settings. * *

All supported hints and options should be added with the Add methods. *
Unknown options will be errors. *
Unknown hints without qualifiers, or with {@code qualifier}, will be * errors. *
Unkonwn hints with other qualifiers will be allowed (because these are * typically interpreted as hints intended for other engines). */ public AllowedHintsAndOptions(String qualifier) { disallowUnknownOptions = true; disallowUnknownHintsWithQualifier(qualifier); disallowUnknownHintsWithQualifier(""); } /** * Serialize this AllowedHintsAndOptions into protobuf, with * FileDescriptors emitted to the builder as needed. */ AllowedHintsAndOptionsProto serialize( FileDescriptorSetsBuilder fileDescriptorSetsBuilder) { AllowedHintsAndOptionsProto.Builder builder = AllowedHintsAndOptionsProto.newBuilder(); builder.setDisallowUnknownOptions(disallowUnknownOptions); builder.addAllDisallowUnknownHintsWithQualifier(disallowUnknownHintsWithQualifiers.values()); for (Hint hint : hints.values()) { if (hint.getQualifier().isEmpty() && !hint.getAllowUnqualified()) { continue; } if (hint.getType() != null) { TypeProto.Builder typeBuilder = TypeProto.newBuilder(); hint.getType().serialize(typeBuilder, fileDescriptorSetsBuilder); builder.addHintBuilder() .setQualifier(hint.getQualifier()).setName(hint.getName()) .setType(typeBuilder).setAllowUnqualified(hint.getAllowUnqualified()); } else { builder.addHintBuilder() .setQualifier(hint.getQualifier()).setName(hint.getName()) .setAllowUnqualified(hint.getAllowUnqualified()); } } for (Entry option : options.entrySet()) { if (option.getValue() != null) { TypeProto.Builder typeBuilder = TypeProto.newBuilder(); option.getValue().serialize(typeBuilder, fileDescriptorSetsBuilder); builder.addOptionBuilder().setName(option.getKey()).setType(typeBuilder); } else { builder.addOptionBuilder().setName(option.getKey()); } } return builder.build(); } /** * Deserialize an AllowedHintsAndOptions from proto. Types will be created * using given type factory and descriptor pools. */ static AllowedHintsAndOptions deserialize( AllowedHintsAndOptionsProto proto, List pools, TypeFactory factory) { AllowedHintsAndOptions allowed = new AllowedHintsAndOptions(); for (String qualifier : proto.getDisallowUnknownHintsWithQualifierList()) { Preconditions.checkArgument( !allowed.disallowUnknownHintsWithQualifiers.containsKey(qualifier.toLowerCase())); allowed.disallowUnknownHintsWithQualifier(qualifier); } allowed.setDisallowUnknownOptions(proto.getDisallowUnknownOptions()); for (HintProto hint : proto.getHintList()) { if (hint.hasType()) { allowed.addHint(hint.getQualifier(), hint.getName(), factory.deserialize(hint.getType(), pools), hint.getAllowUnqualified()); } else { allowed.addHint(hint.getQualifier(), hint.getName(), null, hint.getAllowUnqualified()); } } for (OptionProto option : proto.getOptionList()) { if (option.hasType()) { allowed.addOption(option.getName(), factory.deserialize(option.getType(), pools)); } else { allowed.addOption(option.getName(), null); } } return allowed; } /** * Add an option. * @param name * @param type may be NULL to indicate that all Types are allowed. */ public void addOption(String name, @Nullable Type type) { Preconditions.checkNotNull(name); Preconditions.checkArgument(!name.isEmpty()); Preconditions.checkArgument(!options.containsKey(name.toLowerCase())); options.put(name.toLowerCase(), type); } /** * Add a hint. * @param qualifier may be empty to add this hint only unqualified, but hints * for some engine should normally allow the engine name as a qualifier. * @param name * @param type may be NULL to indicate that all Types are allowed. * @param allowUnqualified if true, this hint is allowed both unqualified and * qualified with {@code qualifier}. */ public void addHint( String qualifier, String name, @Nullable Type type, boolean allowUnqualified) { Preconditions.checkNotNull(name); Preconditions.checkArgument(!name.isEmpty()); Preconditions.checkNotNull(qualifier); Preconditions.checkArgument(!qualifier.isEmpty() || allowUnqualified, "Cannot have hint with no qualifier and !allowUnqualified"); Preconditions.checkArgument( !hints.contains(qualifier.toLowerCase(), name.toLowerCase())); hints.put(qualifier.toLowerCase(), name.toLowerCase(), new Hint(qualifier, name, type, allowUnqualified)); if (allowUnqualified && !qualifier.isEmpty()) { Preconditions.checkArgument(!hints.contains("", name.toLowerCase())); hints.put("", name.toLowerCase(), new Hint("", name, type, false /* This is to mark this hint is not explicitly added and won't be serialized */)); } } public void setDisallowUnknownOptions(boolean disallowUnknownOptions) { this.disallowUnknownOptions = disallowUnknownOptions; } public boolean getDisallowUnknownOptions() { return disallowUnknownOptions; } public Hint getHint(String qualifier, String name) { return hints.get(qualifier.toLowerCase(), name.toLowerCase()); } public ImmutableList getHintList() { return ImmutableList.copyOf(hints.values()); } public Type getOptionType(String name) { return options.get(name.toLowerCase()); } public ImmutableList getOptionNameList() { return ImmutableList.copyOf(options.keySet()); } public void disallowUnknownHintsWithQualifier(String qualifier) { Preconditions.checkNotNull(qualifier); Preconditions.checkArgument( !disallowUnknownHintsWithQualifiers.containsKey(qualifier.toLowerCase())); disallowUnknownHintsWithQualifiers.put(qualifier.toLowerCase(), qualifier); } public ImmutableList getDisallowUnknownHintsWithQualifiers() { return ImmutableList.copyOf(disallowUnknownHintsWithQualifiers.values()); } /** This class specifies all information of a hint. */ public static class Hint implements Serializable { private String qualifier; private String name; private Type type; private boolean allowUnqualified; Hint(String qualifier, String name, @Nullable Type type, boolean allowUnqualified) { this.qualifier = qualifier; this.name = name; this.type = type; this.allowUnqualified = allowUnqualified; } public String getQualifier() { return qualifier; } public String getName() { return name; } @Nullable public Type getType() { return type; } /** * AllowUnqualified is only used when adding new hints and serializing to * proto to recreate hints in C++ side. This should only be used inside the * class and should not be used by users. */ private boolean getAllowUnqualified() { return allowUnqualified; } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy