com.google.zetasql.PreparedModify Maven / Gradle / Ivy
/*
* Copyright 2019 Google LLC
*
* 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 static com.google.zetasql.Parameter.normalize;
import static com.google.zetasql.Parameter.normalizeAndValidate;
import static com.google.zetasql.Parameter.serialize;
import com.google.common.base.Preconditions;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableMap;
import com.google.errorprone.annotations.CanIgnoreReturnValue;
import com.google.errorprone.annotations.CheckReturnValue;
import com.google.zetasql.LocalService.EvaluateModifyRequest;
import com.google.zetasql.LocalService.EvaluateModifyResponse;
import com.google.zetasql.LocalService.PrepareModifyRequest;
import com.google.zetasql.LocalService.PrepareModifyResponse;
import com.google.zetasql.LocalService.PreparedModifyState;
import com.google.zetasql.LocalService.UnprepareModifyRequest;
import io.grpc.StatusRuntimeException;
import java.util.Map;
/**
* ZetaSQL modify statement evaluation using Service RPC.
*
* A prepared modify will create server side state, which should be released by calling close()
* when the statement is no longer used. Note that the close() method is called in finalize(), but
* you should not rely on that because the Java garbage collector has no idea how much memory is
* used on C++ side and GC may happen later than necessary.
*
*
Read the unit tests for examples.
*
*
This class is thread-safe.
*/
@CheckReturnValue
public final class PreparedModify implements AutoCloseable {
private boolean closed = false;
private final long preparedModifyId;
// expected are the fields we expect the user to provide to execute, but not all are required.
private final ImmutableMap expectedParameters;
// referenced is the subset of expected that the analyzer determined are actually required.
private final ImmutableList referencedParameters;
private final SimpleCatalog catalog;
private PreparedModify(
long preparedModifyId,
ImmutableMap expectedParameters,
ImmutableList referencedParameters,
SimpleCatalog catalog) {
this.preparedModifyId = preparedModifyId;
this.expectedParameters = expectedParameters;
this.referencedParameters = referencedParameters;
this.catalog = catalog;
}
/**
* Get the list of parameters referenced in this statement.
*
* The parameters will be returned in lower case, as parameters are case-insensitive. The list
* of parameters returned from this method are the minimal set that must be provided to execute().
*/
public ImmutableList getReferencedParameters() {
return referencedParameters;
}
/**
* Evaluate the SQL statement via Service RPC.
*
* NullPointerException is being thrown if parameters is null
*
*
IllegalStatementException is being thrown if the prepare modify is closed.
*
* @param parameters Map of parameter name:value pairs used in the SQL expression.
* @return The result in the form of a table content.
*/
public EvaluatorTableModifyResponse execute(Map parameters) {
Preconditions.checkNotNull(parameters);
Preconditions.checkState(!closed);
EvaluateModifyRequest request = buildEvaluateRequest(parameters);
try {
final EvaluateModifyResponse response = Client.getStub().evaluateModify(request);
return EvaluatorTableModifyResponse.deserialize(this.catalog, response);
} catch (StatusRuntimeException e) {
throw new SqlException(e);
}
}
private EvaluateModifyRequest buildEvaluateRequest(Map parameters) {
EvaluateModifyRequest.Builder requestBuilder = EvaluateModifyRequest.newBuilder();
requestBuilder.setPreparedModifyId(preparedModifyId);
final ImmutableMap normalizedParameters =
normalizeAndValidate(parameters, expectedParameters, "query");
for (String param : referencedParameters) {
Value value = normalizedParameters.get(param);
if (value == null) {
throw new SqlException("Incomplete sql expression parameters " + param);
}
requestBuilder.addParams(serialize(param, value));
}
return requestBuilder.build();
}
/** Release the server side state for this prepared modify statement. */
@Override
public void close() {
if (closed) {
return;
}
try {
UnprepareModifyRequest request =
UnprepareModifyRequest.newBuilder().setPreparedModifyId(preparedModifyId).build();
Client.getStub().unprepareModify(request);
} catch (StatusRuntimeException e) {
// ignore
}
closed = true;
}
@Override
protected void finalize() throws Throwable {
super.finalize();
close();
}
/** Get a new builder. */
public static Builder builder() {
return new PreparedModify.Builder();
}
/**
* A builder for creating immutable PreparedModify instances. Example:
*
* {@code
* public final PreparedModify modify =
* PreparedModify.builder()
* .setSql(sqlString)
* .setAnalyzerOptions(options)
* .setCatalog(catalog)
* .prepare();
*
* public final PreparedModify modifyWithTablesContents =
* PreparedModify.builder()
* .setSql(sqlString)
* .setAnalyzerOptions(options)
* .setCatalog(catalog)
* .setTablesContents(tablesContents)
* .prepare();
* }
*
* Builder instances can be reused; it is safe to call {@link #prepare} multiple times to
* create multiple instances of PreparedModify. Every time {@link #prepare} is called a new modify
* is registered with the local service and a new instance is returned.
*/
public static final class Builder {
private String sql;
private AnalyzerOptions options;
private SimpleCatalog catalog;
private Map tablesContents;
private Builder() {}
/**
* Set the string SQL statement.
*
* Mandatory.
*/
@CanIgnoreReturnValue
public Builder setSql(String sql) {
this.sql = sql;
return this;
}
/**
* Set the options to customize the SQL statement's analyzer behavior.
*
*
Mandatory.
*/
@CanIgnoreReturnValue
public Builder setAnalyzerOptions(AnalyzerOptions options) {
this.options = options;
return this;
}
/**
* Set the catalog with which this SQL statement interacts.
*
*
Mandatory.
*/
@CanIgnoreReturnValue
public Builder setCatalog(SimpleCatalog catalog) {
this.catalog = catalog;
return this;
}
/**
* Set the content for the tables within the catalog.
*
*
Optional.
*
*
It can be used only together with a catalog that is not registered.
*/
@CanIgnoreReturnValue
public Builder setTablesContents(Map tablesContents) {
this.tablesContents = tablesContents;
return this;
}
/**
* Prepare the SQL statement and return a new instance of PreparedModify. Throwing SqlException
* if there's an error (not necessarily network/server failure).
*
* IllegalStateException is being thrown if the SQL string is null
*
*
IllegalStateException is being thrown if the options is null
*
*
IllegalStateException is being thrown if the catalog is null
*
*
IllegalStateException is being thrown if the provided catalog is a registered catalog and
* tables contents were provided. At the moment changing the contents of the tables of a
* registered catalog is not supported yet.
*/
public PreparedModify prepare() {
Preconditions.checkState(sql != null, "SQL string must not be null");
Preconditions.checkState(options != null, "AnalyzerOptions must not be null");
Preconditions.checkState(catalog != null, "Catalog must not be null");
Preconditions.checkState(
tablesContents == null || !catalog.isRegistered(),
"Can only provide tables contents with a catalog that has not been registered");
final PrepareModifyRequest request = buildPrepareRequest();
try {
final PrepareModifyResponse response = Client.getStub().prepareModify(request);
return processResponse(response.getPrepared());
} catch (StatusRuntimeException e) {
throw new SqlException(e);
}
}
private PrepareModifyRequest buildPrepareRequest() {
FileDescriptorSetsBuilder fileDescriptorSetsBuilder = new FileDescriptorSetsBuilder();
fileDescriptorSetsBuilder.addAllFileDescriptors(BuiltinDescriptorPool.getInstance());
PrepareModifyRequest.Builder builder = PrepareModifyRequest.newBuilder();
builder.setSql(sql);
builder.setOptions(options.serialize(fileDescriptorSetsBuilder));
Map catalogRegisteredDescriptorPoolIds = ImmutableMap.of();
if (catalog.isRegistered()) {
catalogRegisteredDescriptorPoolIds = catalog.getRegisteredDescriptorPoolIds();
for (DescriptorPool pool : catalogRegisteredDescriptorPoolIds.keySet()) {
fileDescriptorSetsBuilder.addAllFileDescriptors(pool);
}
builder.setRegisteredCatalogId(catalog.getRegisteredId());
} else {
builder.setSimpleCatalog(catalog.serialize(fileDescriptorSetsBuilder));
if (tablesContents != null) {
tablesContents.forEach((t, c) -> builder.putTableContent(t, c.serialize()));
}
}
builder.setDescriptorPoolList(
DescriptorPoolSerializer.createDescriptorPoolListWithRegisteredIds(
fileDescriptorSetsBuilder, catalogRegisteredDescriptorPoolIds));
return builder.build();
}
private PreparedModify processResponse(PreparedModifyState preparedModifyState) {
long preparedModifyId = preparedModifyState.getPreparedModifyId();
ImmutableMap expectedParameters = normalize(options.getQueryParameters());
ImmutableList referencedParameters =
ImmutableList.copyOf(preparedModifyState.getReferencedParametersList());
return new PreparedModify(
preparedModifyId, expectedParameters, referencedParameters, catalog);
}
}
}