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

com.google.zetasql.SimpleCatalog 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.ImmutableList;
import com.google.protobuf.BoolValue;
import com.google.protobuf.BytesValue;
import com.google.protobuf.DoubleValue;
import com.google.protobuf.FloatValue;
import com.google.protobuf.Int32Value;
import com.google.protobuf.Int64Value;
import com.google.protobuf.StringValue;
import com.google.protobuf.Timestamp;
import com.google.protobuf.UInt32Value;
import com.google.protobuf.UInt64Value;
import com.google.zetasql.FunctionProtos.FunctionProto;
import com.google.zetasql.FunctionProtos.ProcedureProto;
import com.google.zetasql.FunctionProtos.TableValuedFunctionProto;
import com.google.zetasql.ZetaSQLDescriptorPool.ZetaSQLDescriptor;
import com.google.zetasql.ZetaSQLDescriptorPool.ZetaSQLEnumDescriptor;
import com.google.zetasql.ZetaSQLOptionsProto.ZetaSQLBuiltinFunctionOptionsProto;
import com.google.zetasql.ZetaSQLType.TypeProto;
import com.google.zetasql.LocalService.AddSimpleTableRequest;
import com.google.zetasql.LocalService.GetBuiltinFunctionsResponse;
import com.google.zetasql.LocalService.RegisterCatalogRequest;
import com.google.zetasql.LocalService.RegisterResponse;
import com.google.zetasql.LocalService.UnregisterRequest;
import com.google.zetasql.SimpleCatalogProtos.SimpleCatalogProto;
import com.google.zetasql.SimpleCatalogProtos.SimpleCatalogProto.NamedTypeProto;
import com.google.zetasql.SimpleConstantProtos.SimpleConstantProto;
import com.google.zetasql.SimpleTableProtos.SimpleTableProto;
import com.google.zetasql.functions.ZetaSQLDateTime.DateTimestampPart;
import com.google.zetasql.functions.ZetaSQLNormalizeMode.NormalizeMode;
import com.google.type.Date;
import com.google.type.LatLng;
import com.google.type.TimeOfDay;
import io.grpc.StatusRuntimeException;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
import java.util.logging.Logger;

/**
 * SimpleCatalog is a concrete implementation of the Catalog interface. It acts as a simple
 * container for objects in the Catalog.
 *
 * 

This class and related SimpleX classes are final by design because a subclass is unlikely to * get serialize method right. The serialize method returns a SimpleCatalogProto. A subclass most * likely needs more fields and hence needs a different proto. But proto class is final so a * subclass cannot override this class's serialize method and return a subclass of * SimpleCatalogProto. TODO: Add findTableById. * *

To improve RPC performance, a SimpleCatalog can be registered on the local server, and later * requests involving this SimpleCatalog will only pass the registered ID instead of the fully * serialized form. A registered SimpleCatalog can and should be unregistered when it is no longer * used. The unregister() method is called in finalize(), but users shouldn't rely on that because * the JVM doesn't know of the memory resource taken by the registered catalog on the local * server, and may start the garbage collector later than necessary. */ public class SimpleCatalog extends Catalog { private static final Logger logger = Logger.getLogger(SimpleCatalog.class.getName()); private final String name; private final TypeFactory typeFactory; // All String keys are stored in lower case. private final Map constants = new HashMap<>(); private final Map tables = new HashMap<>(); private final Map types = new HashMap<>(); private final Map catalogs = new HashMap<>(); private final Map tablesById = new HashMap<>(); private final Map customFunctions = new HashMap<>(); private final Map tvfs = new HashMap<>(); private final Map functionsByFullName = new HashMap<>(); private final Map procedures = new HashMap<>(); private ZetaSQLDescriptorPool descriptorPool; ZetaSQLBuiltinFunctionOptionsProto builtinFunctionOptions; boolean registered = false; long registeredId = -1; FileDescriptorSetsBuilder registeredFileDescriptorSetsBuilder = null; public SimpleCatalog(String name, TypeFactory typeFactory) { this.name = name; this.typeFactory = Preconditions.checkNotNull(typeFactory); } public SimpleCatalog(String name) { this(name, TypeFactory.nonUniqueNames()); } /** * Register this catalog to local server, so that it can be reused without passing through RPC * every time. * * @return An AutoCloseable object that can be used to unregister the catalog in * try-with-resources structure. */ public AutoUnregister register() { Preconditions.checkState(!registered); registeredFileDescriptorSetsBuilder = new FileDescriptorSetsBuilder(); RegisterCatalogRequest.Builder builder = RegisterCatalogRequest.newBuilder(); builder.setSimpleCatalog(serialize(registeredFileDescriptorSetsBuilder)); builder.addAllFileDescriptorSet(registeredFileDescriptorSetsBuilder.build()); try { RegisterResponse resp = Client.getStub().registerCatalog(builder.build()); registeredId = resp.getRegisteredId(); } catch (StatusRuntimeException e) { throw new SqlException(e); } registered = true; return new AutoUnregister(); } public void unregister() { Preconditions.checkState(registered); try { Client.getStub() .unregisterCatalog(UnregisterRequest.newBuilder().setRegisteredId(registeredId).build()); } catch (StatusRuntimeException e) { // Maybe caused by double unregistering (in race conditions) or RPC // failure. The latter may cause leak but it's likely due to more serious // problems and there is no good way to recover. Just log and ignore. logger.severe("Failed to unregister catalog: " + e.getMessage()); } finally { registered = false; registeredId = -1; registeredFileDescriptorSetsBuilder = null; } } public boolean isRegistered() { return registered; } long getRegisteredId() { Preconditions.checkState(registered); return registeredId; } @Override protected void finalize() { if (registered) { unregister(); } } FileDescriptorSetsBuilder getRegisteredFileDescriptorSetsBuilder() { Preconditions.checkState(registered); return registeredFileDescriptorSetsBuilder; } /** * Serialize this catalog into protobuf, with FileDescriptors emitted to the builder as needed. */ public SimpleCatalogProto serialize(FileDescriptorSetsBuilder fileDescriptorSetsBuilder) { SimpleCatalogProto.Builder builder = SimpleCatalogProto.newBuilder(); builder.setName(name); // The built-in function definations are not serialized. Instead, the BuiltinFunctionOptions // which specify which functions to include and exclude will be serialized, and the C++ // deserialization will recreate the same built-in function signatures according to this. if (builtinFunctionOptions != null) { builder.setBuiltinFunctionOptions(builtinFunctionOptions); } for (Entry table : tables.entrySet()) { SimpleTableProto.Builder tableBuilder = builder.addTableBuilder(); tableBuilder.mergeFrom(table.getValue().serialize(fileDescriptorSetsBuilder)); if (!table.getKey().equals(tableBuilder.getName().toLowerCase())) { tableBuilder.setNameInCatalog(table.getKey()); } } for (Entry type : types.entrySet()) { TypeProto.Builder typeProtoBuilder = TypeProto.newBuilder(); type.getValue().serialize(typeProtoBuilder, fileDescriptorSetsBuilder); builder.addNamedType( NamedTypeProto.newBuilder().setName(type.getKey()).setType(typeProtoBuilder.build())); } for (Entry catalog : catalogs.entrySet()) { builder.addCatalog(catalog.getValue().serialize(fileDescriptorSetsBuilder)); } for (Entry tvf : tvfs.entrySet()) { builder.addCustomTvf(tvf.getValue().serialize(fileDescriptorSetsBuilder)); } // TODO: Handle serialization of ResolvedFunctionCallInfo. // TODO: Handle serialization of TVFSignatures. for (Entry constant : constants.entrySet()) { builder.addConstant(constant.getValue().serialize(fileDescriptorSetsBuilder)); } for (Entry function : customFunctions.entrySet()) { builder.addCustomFunction(function.getValue().serialize(fileDescriptorSetsBuilder)); } for (Entry procedure : procedures.entrySet()) { builder.addProcedure(procedure.getValue().serialize(fileDescriptorSetsBuilder)); } if (descriptorPool != null) { builder.setFileDescriptorSetIndex( fileDescriptorSetsBuilder.addAllFileDescriptors(descriptorPool)); } return builder.build(); } /** Add simple constant into this catalog. Constant names are case insensitive. */ public void addConstant(Constant constant) { Preconditions.checkState(!registered); Preconditions.checkArgument(!constants.containsKey(constant.getFullName().toLowerCase())); List namePath = constant.getNamePath(); Preconditions.checkArgument(!namePath.isEmpty()); constants.put(namePath.get(namePath.size() - 1).toLowerCase(), constant); } /** * Add simple table into this catalog. Table names are case insensitive. * * @param table */ public void addSimpleTable(SimpleTable table) { addSimpleTable(table.getName(), table); } public void addSimpleTable(String name, SimpleTable table) { Preconditions.checkArgument(!tables.containsKey(name.toLowerCase()), "duplicate key: %s", name); if (registered) { synchronized (this) { try { FileDescriptorSetsBuilder temp = new FileDescriptorSetsBuilder(); temp.importDescriptorPoolIndex(registeredFileDescriptorSetsBuilder); SimpleTableProto tableProto = table.serialize(temp); FileDescriptorSetsBuilder diff = registeredFileDescriptorSetsBuilder.mergeDiff(temp); AddSimpleTableRequest.Builder requestBuilder = AddSimpleTableRequest.newBuilder(); requestBuilder.setRegisteredCatalogId(registeredId); requestBuilder.setTable(tableProto); requestBuilder.addAllFileDescriptorSet(diff.build()); Client.getStub().addSimpleTable(requestBuilder.build()); } catch (StatusRuntimeException e) { throw new SqlException(e); } } } tables.put(name.toLowerCase(), table); tablesById.put(table.getId(), table); } /** * Removes a simple table from this catalog. */ public void removeSimpleTable(SimpleTable table) { removeSimpleTable(table.getName()); } /** * Removes a simple table from this catalog. Table names are case insensitive. */ public void removeSimpleTable(String name) { Preconditions.checkState(!registered); Preconditions.checkArgument(tables.containsKey(name.toLowerCase()), "missing key: %s", name); SimpleTable table = tables.remove(name.toLowerCase()); tablesById.remove(table.getId()); } /** * Add type into this catalog. Type names are case insensitive. * * @param name * @param type */ public void addType(String name, Type type) { Preconditions.checkState(!registered); Preconditions.checkArgument(!types.containsKey(name.toLowerCase()), "duplicate key: %s", name); types.put(name.toLowerCase(), type); } /** * Removes the type with the passed name from this catalog. Type names are case insensitive. */ public void removeType(String name) { Preconditions.checkState(!registered); Preconditions.checkArgument(types.containsKey(name.toLowerCase()), "missing key: %s", name); types.remove(name.toLowerCase()); } /** * Add sub catalog into this catalog. Catalog names are case insensitive. * * @param catalog */ public void addSimpleCatalog(SimpleCatalog catalog) { Preconditions.checkState(!registered); Preconditions.checkArgument( !catalogs.containsKey(catalog.getFullName().toLowerCase()), "duplicate key: %s", catalog.getFullName()); catalogs.put(catalog.getFullName().toLowerCase(), catalog); } /** * Create a new catalog and add the new catalog into this catalog and return the new sub catalog. * Catalog names are case insensitive. * * @param name * @return the created new sub catalog */ public SimpleCatalog addNewSimpleCatalog(String name) { Preconditions.checkState(!registered); SimpleCatalog newCatalog = new SimpleCatalog(name, getTypeFactory()); addSimpleCatalog(newCatalog); return newCatalog; } /** * Removes the passed sub catalog from this catalog. */ public void removeSimpleCatalog(SimpleCatalog catalog) { removeSimpleCatalog(catalog.getFullName()); } /** * Removes the sub catalog with the passed name from this catalog. Catalog names are case * insensitive. */ public void removeSimpleCatalog(String fullName) { Preconditions.checkState(!registered); Preconditions.checkArgument(catalogs.containsKey(fullName.toLowerCase()), "missing key: %s", fullName); catalogs.remove(fullName.toLowerCase()); } /** * Add ZetaSQL built-in function definitions into this catalog. Function names are case * insensitive. * * @param options used to select which functions get loaded. */ public void addZetaSQLFunctions(ZetaSQLBuiltinFunctionOptions options) { Preconditions.checkNotNull(options); Preconditions.checkState(builtinFunctionOptions == null); builtinFunctionOptions = options.serialize(); try { GetBuiltinFunctionsResponse response = Client.getStub().getBuiltinFunctions(builtinFunctionOptions); processGetBuiltinFunctionsResponse(response); } catch (StatusRuntimeException e) { throw new SqlException(e); } } void processGetBuiltinFunctionsResponse(GetBuiltinFunctionsResponse response) { // Datetime and string normalization functions have arguments of enum types, which needs to // import the enum descriptor to descriptor pool so that these argument types can be // deserialized properly. // This should be updated when new built in functions with enum or proto type arguments // are added. ZetaSQLDescriptorPool.importIntoGeneratedPool(DateTimestampPart.getDescriptor()); ZetaSQLDescriptorPool.importIntoGeneratedPool(NormalizeMode.getDescriptor()); ZetaSQLDescriptorPool.importIntoGeneratedPool(Timestamp.getDescriptor()); ZetaSQLDescriptorPool.importIntoGeneratedPool(Date.getDescriptor()); ZetaSQLDescriptorPool.importIntoGeneratedPool(TimeOfDay.getDescriptor()); ZetaSQLDescriptorPool.importIntoGeneratedPool(LatLng.getDescriptor()); ZetaSQLDescriptorPool.importIntoGeneratedPool(DoubleValue.getDescriptor()); ZetaSQLDescriptorPool.importIntoGeneratedPool(FloatValue.getDescriptor()); ZetaSQLDescriptorPool.importIntoGeneratedPool(Int64Value.getDescriptor()); ZetaSQLDescriptorPool.importIntoGeneratedPool(UInt64Value.getDescriptor()); ZetaSQLDescriptorPool.importIntoGeneratedPool(Int32Value.getDescriptor()); ZetaSQLDescriptorPool.importIntoGeneratedPool(UInt32Value.getDescriptor()); ZetaSQLDescriptorPool.importIntoGeneratedPool(BoolValue.getDescriptor()); ZetaSQLDescriptorPool.importIntoGeneratedPool(StringValue.getDescriptor()); ZetaSQLDescriptorPool.importIntoGeneratedPool(BytesValue.getDescriptor()); ImmutableList pools = ImmutableList.of(ZetaSQLDescriptorPool.getGeneratedPool()); for (FunctionProto proto : response.getFunctionList()) { addFunctionToFullNameMap(Function.deserialize(proto, pools)); } } /** * Add function into this catalog. Function name and alias name will both be stored if different. * Names are case insensitive. * * @param function */ public void addFunction(Function function) { Preconditions.checkState(!registered); Preconditions.checkArgument(!customFunctions.containsKey(function.getName().toLowerCase())); customFunctions.put(function.getName().toLowerCase(), function); if (!function.getOptions().getAliasName().isEmpty() && !function.getOptions().getAliasName().equals(function.getName())) { Preconditions.checkArgument( !customFunctions.containsKey(function.getOptions().getAliasName().toLowerCase())); customFunctions.put(function.getOptions().getAliasName().toLowerCase(), function); } addFunctionToFullNameMap(function); } private void addFunctionToFullNameMap(Function function) { Preconditions.checkArgument( !functionsByFullName.containsKey(function.getFullName().toLowerCase())); functionsByFullName.put(function.getFullName().toLowerCase(), function); } /** * Removes the passed function from this catalog. */ public void removeFunction(Function function) { removeFunction(function.getFullName()); } /** * Removes the function with the passed full name from this catalog. */ public void removeFunction(String fullName) { Preconditions.checkState(!registered); Function function = getFunctionByFullName(fullName); Preconditions.checkArgument(function != null); Preconditions.checkArgument(customFunctions.containsKey(function.getName().toLowerCase())); customFunctions.remove(function.getName().toLowerCase()); if (!function.getOptions().getAliasName().isEmpty()) { customFunctions.remove(function.getOptions().getAliasName().toLowerCase()); } removeFunctionFromFullNameMap(function); } /** Add the given {@code tvf} to this catalog. Names are case insensitive. */ public void addTableValuedFunction(TableValuedFunction tvf) { Preconditions.checkState(!registered); String tvfName = tvf.getName().toLowerCase(); Preconditions.checkArgument(!tvfs.containsKey(tvfName)); tvfs.put(tvfName, tvf); } /** Removes the passed TVF from this catalog. */ public void removeTableValuedFunction(TableValuedFunction tvf) { removeTableValuedFunction(tvf.getFullName()); } /** Removes the TVF with the passed full name from this catalog. */ public void removeTableValuedFunction(String name) { Preconditions.checkState(!registered); String lowerCaseName = name.toLowerCase(); TableValuedFunction tvf = getTVFByName(lowerCaseName); Preconditions.checkArgument(tvf != null); tvfs.remove(lowerCaseName); } private void removeFunctionFromFullNameMap(Function function) { Preconditions.checkArgument( functionsByFullName.containsKey(function.getFullName().toLowerCase())); functionsByFullName.remove(function.getFullName().toLowerCase()); } /** * Add procedure into this catalog. Names are case insensitive. * * @param procedure */ public void addProcedure(Procedure procedure) { Preconditions.checkState(!registered); Preconditions.checkArgument(!procedures.containsKey(procedure.getName().toLowerCase())); procedures.put(procedure.getName().toLowerCase(), procedure); } /** * Removes the passed procedure from this catalog. */ public void removeProcedure(Procedure procedure) { removeProcedure(procedure.getName()); } /** * Removes the procedure with the passed name from this catalog. Names are case insensitive. */ public void removeProcedure(String name) { Preconditions.checkState(!registered); Preconditions.checkArgument(procedures.containsKey(name.toLowerCase())); procedures.remove(name.toLowerCase()); } public void setDescriptorPool(ZetaSQLDescriptorPool descriptorPool) { Preconditions.checkState(!registered); Preconditions.checkState(this.descriptorPool == null); this.descriptorPool = descriptorPool; } @Override public String getFullName() { return name; } public TypeFactory getTypeFactory() { return typeFactory; } public ImmutableList getTableNameList() { return ImmutableList.copyOf(tables.keySet()); } public ImmutableList getTableList() { return ImmutableList.copyOf(tables.values()); } public ImmutableList getTypeList() { return ImmutableList.copyOf(types.values()); } public ImmutableList getCatalogList() { return ImmutableList.copyOf(catalogs.values()); } public ImmutableList getFunctionList() { return ImmutableList.copyOf(functionsByFullName.values()); } public ImmutableList getTVFList() { return ImmutableList.copyOf(tvfs.values()); } public ImmutableList getFunctionNameList() { return ImmutableList.copyOf(functionsByFullName.keySet()); } public ImmutableList getTVFNameList() { return ImmutableList.copyOf(tvfs.keySet()); } public ImmutableList getProcedureList() { return ImmutableList.copyOf(procedures.values()); } @Override protected Constant getConstant(String name, FindOptions options) { return constants.get(name.toLowerCase()); } @Override public SimpleTable getTable(String name, FindOptions options) { return tables.get(name.toLowerCase()); } @Override public Type getType(String name, FindOptions options) { if (types.containsKey(name.toLowerCase())) { return types.get(name.toLowerCase()); } if (descriptorPool != null) { ZetaSQLDescriptor descriptor = descriptorPool.findMessageTypeByName(name); if (descriptor != null) { return getTypeFactory().createProtoType(descriptor); } ZetaSQLEnumDescriptor enumDescriptor = descriptorPool.findEnumTypeByName(name); if (enumDescriptor != null) { return getTypeFactory().createEnumType(enumDescriptor); } } return null; } @Override public SimpleCatalog getCatalog(String name, FindOptions options) { return catalogs.get(name.toLowerCase()); } public Function getFunctionByFullName(String fullName) { Function function = functionsByFullName.get(fullName.toLowerCase()); if (function == null) { for (SimpleCatalog catalog : catalogs.values()) { function = catalog.getFunctionByFullName(fullName); if (function != null) { break; } } } return function; } public TableValuedFunction getTVFByName(String name) { return tvfs.get(name.toLowerCase()); } public SimpleTable getTableById(long serializationId) { SimpleTable table = tablesById.get(serializationId); if (table == null) { for (SimpleCatalog catalog : catalogs.values()) { table = catalog.getTableById(serializationId); if (table != null) { break; } } } return table; } public SimpleModel getModelById(long serializationId) { // TODO: Add support for Model in the Java implementation. return null; } public SimpleConnection getConnectionByFullName(String fullName) { // TODO: Add support for Connection in the Java implementation. return null; } @Override protected Procedure getProcedure(String name, FindOptions options) { return procedures.get(name.toLowerCase()); } /** * AutoCloseable implementation that unregisters the SimpleCatalog automatically when used in * try-with-resources. */ public class AutoUnregister implements AutoCloseable { @Override public void close() { unregister(); } } public static SimpleCatalog deserialize( SimpleCatalogProto proto, ImmutableList pools) { SimpleCatalog catalog = new SimpleCatalog(proto.getName()); for (SimpleTableProto tableProto : proto.getTableList()) { String name = tableProto.hasNameInCatalog() ? tableProto.getNameInCatalog() : tableProto.getName(); catalog.addSimpleTable( name, SimpleTable.deserialize(tableProto, pools, catalog.getTypeFactory())); } for (NamedTypeProto typeProto : proto.getNamedTypeList()) { catalog.addType( typeProto.getName(), catalog.getTypeFactory().deserialize(typeProto.getType(), pools)); } for (SimpleCatalogProto catalogProto : proto.getCatalogList()) { catalog.addSimpleCatalog(SimpleCatalog.deserialize(catalogProto, pools)); } // TODO: Handle deserialization of ResolvedFunctionCallInfo. // TODO: Handle deserialization of TVFSignatures. for (SimpleConstantProto constantProto : proto.getConstantList()) { catalog.addConstant(Constant.deserialize(constantProto, pools, catalog.getTypeFactory())); } for (TableValuedFunctionProto tvfProto : proto.getCustomTvfList()) { catalog.addTableValuedFunction( TableValuedFunction.deserialize(tvfProto, pools, catalog.getTypeFactory())); } for (FunctionProto functionProto : proto.getCustomFunctionList()) { catalog.addFunction(Function.deserialize(functionProto, pools)); } for (ProcedureProto procedureProto : proto.getProcedureList()) { catalog.addProcedure(Procedure.deserialize(procedureProto, pools)); } if (proto.hasBuiltinFunctionOptions()) { ZetaSQLBuiltinFunctionOptions options = new ZetaSQLBuiltinFunctionOptions(proto.getBuiltinFunctionOptions()); catalog.addZetaSQLFunctions(options); } if (proto.hasFileDescriptorSetIndex()) { catalog.setDescriptorPool(pools.get(proto.getFileDescriptorSetIndex())); } return catalog; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy