io.r2dbc.postgresql.SimpleQueryPostgresqlStatement Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of r2dbc-postgresql Show documentation
Show all versions of r2dbc-postgresql Show documentation
Reactive Relational Database Connectivity Driver Implementation for Postgresql
/*
* Copyright 2017 the original author or 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
*
* https://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 io.r2dbc.postgresql;
import io.r2dbc.postgresql.api.PostgresqlStatement;
import io.r2dbc.postgresql.client.Binding;
import io.r2dbc.postgresql.client.SimpleQueryMessageFlow;
import io.r2dbc.postgresql.message.backend.BackendMessage;
import io.r2dbc.postgresql.message.backend.CommandComplete;
import io.r2dbc.postgresql.message.backend.EmptyQueryResponse;
import io.r2dbc.postgresql.message.backend.ErrorResponse;
import io.r2dbc.postgresql.util.Assert;
import io.r2dbc.postgresql.util.GeneratedValuesUtils;
import io.r2dbc.postgresql.util.Operators;
import io.r2dbc.spi.Statement;
import reactor.core.publisher.Flux;
import reactor.util.annotation.Nullable;
import java.util.Collections;
import java.util.function.Predicate;
import static io.r2dbc.postgresql.message.frontend.Execute.NO_LIMIT;
import static io.r2dbc.postgresql.util.PredicateUtils.or;
/**
* {@link Statement} using the Simple Query Flow. Can use the {@link ExtendedQueryPostgresqlStatement extended
* flow} when a {@link #fetchSize(int) fetch size} is specified.
*/
final class SimpleQueryPostgresqlStatement implements PostgresqlStatement {
private static final boolean REACTOR_3_4_AVAILABLE = isPresent("reactor.util.context.ContextView", SimpleQueryPostgresqlStatement.class.getClassLoader());
private static final Predicate WINDOW_UNTIL = or(CommandComplete.class::isInstance, EmptyQueryResponse.class::isInstance, ErrorResponse.class::isInstance);
private final ConnectionResources resources;
private final String sql;
private String[] generatedColumns;
private int fetchSize;
SimpleQueryPostgresqlStatement(ConnectionResources resources, String sql) {
this.resources = Assert.requireNonNull(resources, "resources must not be null");
this.sql = Assert.requireNonNull(sql, "sql must not be null");
fetchSize(isBatch() ? NO_LIMIT : this.resources.getConfiguration().getFetchSize(sql));
}
private boolean isBatch() {
return this.sql.contains(";");
}
@Override
public SimpleQueryPostgresqlStatement add() {
return this;
}
@Override
public SimpleQueryPostgresqlStatement bind(@Nullable String identifier, @Nullable Object value) {
throw new UnsupportedOperationException(String.format("Binding parameters is not supported for the statement '%s'", this.sql));
}
@Override
public SimpleQueryPostgresqlStatement bind(int index, @Nullable Object value) {
throw new UnsupportedOperationException(String.format("Binding parameters is not supported for the statement '%s'", this.sql));
}
@Override
public SimpleQueryPostgresqlStatement bindNull(@Nullable String identifier, @Nullable Class> type) {
throw new UnsupportedOperationException(String.format("Binding parameters is not supported for the statement '%s'", this.sql));
}
@Override
public SimpleQueryPostgresqlStatement bindNull(int index, @Nullable Class> type) {
throw new UnsupportedOperationException(String.format("Binding parameters is not supported for the statement '%s'", this.sql));
}
@Override
public Flux execute() {
return execute(this.generatedColumns == null ? this.sql : GeneratedValuesUtils.augment(this.sql, this.generatedColumns));
}
@Override
public SimpleQueryPostgresqlStatement fetchSize(int rows) {
Assert.isTrue(rows >= 0, "Fetch size must be greater or equal zero");
if (rows != NO_LIMIT) {
Assert.isTrue(!isBatch(), "Fetch size can only be used with a single SQL statement");
}
this.fetchSize = rows;
return this;
}
@Override
public SimpleQueryPostgresqlStatement returnGeneratedValues(String... columns) {
Assert.requireNonNull(columns, "columns must not be null");
if (GeneratedValuesUtils.hasReturningClause(this.sql)) {
throw new IllegalStateException("Statement already includes RETURNING clause");
}
if (!GeneratedValuesUtils.isSupportedCommand(this.sql)) {
throw new IllegalStateException("Statement is not a DELETE, INSERT, or UPDATE command");
}
this.generatedColumns = columns;
return this;
}
@Override
public String toString() {
return "SimpleQueryPostgresqlStatement{" +
"context=" + this.resources +
", sql='" + this.sql + '\'' +
'}';
}
static boolean supports(String sql) {
Assert.requireNonNull(sql, "sql must not be null");
return sql.trim().isEmpty() || !sql.contains("$1");
}
private Flux execute(String sql) {
ExceptionFactory factory = ExceptionFactory.withSql(sql);
if (this.fetchSize != NO_LIMIT) {
Flux messages = ExtendedFlowDelegate.runQuery(this.resources, factory, sql, Binding.EMPTY, Collections.emptyList(), this.fetchSize);
return REACTOR_3_4_AVAILABLE ? messages.windowUntil(WINDOW_UNTIL).map(msg -> new PostgresqlResult(this.resources, msg, factory)).as(Operators::discardOnCancel) :
Flux.just(new PostgresqlResult(this.resources, messages, factory));
}
Flux messages = SimpleQueryMessageFlow.exchange(this.resources.getClient(), sql);
return REACTOR_3_4_AVAILABLE ? messages.windowUntil(WINDOW_UNTIL).map(msg -> new PostgresqlResult(this.resources, msg, factory)).as(Operators::discardOnCancel) :
Flux.just(PostgresqlResult.toResult(this.resources, messages, factory));
}
/**
* Determine whether the {@link Class} identified by the supplied name is present
* and can be loaded. Will return {@code false} if either the class or
* one of its dependencies is not present or cannot be loaded.
*
* @param className the name of the class to check
* @param classLoader the class loader to use
* (may be {@code null} which indicates the default class loader)
* @return whether the specified class is present (including all of its
* superclasses and interfaces)
* @throws IllegalStateException if the corresponding class is resolvable but
* there was a readability mismatch in the inheritance hierarchy of the class
* (typically a missing dependency declaration in a Jigsaw module definition
* for a superclass or interface implemented by the class to be checked here)
*/
private static boolean isPresent(String className, @Nullable ClassLoader classLoader) {
try {
try {
Class.forName(className, false, classLoader);
} catch (ClassNotFoundException ex) {
int lastDotIndex = className.lastIndexOf(".");
if (lastDotIndex != -1) {
String innerClassName =
className.substring(0, lastDotIndex) + "$" + className.substring(lastDotIndex + 1);
try {
Class.forName(innerClassName, false, classLoader);
} catch (ClassNotFoundException ex2) {
// Swallow - let original exception get through
}
}
throw ex;
}
return true;
} catch (IllegalAccessError err) {
throw new IllegalStateException("Readability mismatch in inheritance hierarchy of class [" +
className + "]: " + err.getMessage(), err);
} catch (Throwable ex) {
// Typically ClassNotFoundException or NoClassDefFoundError...
return false;
}
}
}