Please wait. This can take some minutes ...
Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance.
Project price only 1 $
You can buy this project and download/modify it how often you want.
io.vertx.sqlclient.impl.PreparedStatementImpl Maven / Gradle / Ivy
/*
* Copyright (C) 2017 Julien Viet
*
* 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 io.vertx.sqlclient.impl;
import io.vertx.core.impl.ContextInternal;
import io.vertx.core.impl.future.PromiseInternal;
import io.vertx.sqlclient.PrepareOptions;
import io.vertx.sqlclient.PreparedQuery;
import io.vertx.sqlclient.impl.command.CloseCursorCommand;
import io.vertx.sqlclient.impl.command.CloseStatementCommand;
import io.vertx.sqlclient.Cursor;
import io.vertx.sqlclient.PreparedStatement;
import io.vertx.sqlclient.SqlResult;
import io.vertx.sqlclient.RowSet;
import io.vertx.sqlclient.RowStream;
import io.vertx.sqlclient.Row;
import io.vertx.sqlclient.Tuple;
import io.vertx.core.*;
import io.vertx.sqlclient.impl.command.PrepareStatementCommand;
import java.util.List;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.function.Function;
import java.util.stream.Collector;
/**
* @author Julien Viet
*/
class PreparedStatementImpl implements PreparedStatement {
static PreparedStatement create(Connection conn, ContextInternal context, io.vertx.sqlclient.impl.PreparedStatement ps, boolean autoCommit) {
return new PreparedStatementImpl(conn, context, ps, autoCommit);
}
static PreparedStatement create(Connection conn, ContextInternal context, PrepareOptions options, String sql, boolean autoCommit) {
return new PreparedStatementImpl(conn, context, sql, options, autoCommit);
}
private final Connection conn;
private final ContextInternal context;
private final String sql;
private final PrepareOptions options;
private Promise promise;
private Future future;
private final boolean autoCommit;
private final AtomicBoolean closed = new AtomicBoolean();
private PreparedStatementImpl(Connection conn, ContextInternal context, io.vertx.sqlclient.impl.PreparedStatement ps, boolean autoCommit) {
this.conn = conn;
this.context = context;
this.sql = null;
this.options = null;
this.promise = null;
this.future = Future.succeededFuture(ps);
this.autoCommit = autoCommit;
}
private PreparedStatementImpl(Connection conn,
ContextInternal context,
String sql,
PrepareOptions options,
boolean autoCommit) {
this.conn = conn;
this.context = context;
this.sql = sql;
this.options = options;
this.promise = Promise.promise();
this.autoCommit = autoCommit;
}
PrepareOptions options() {
return options;
}
@Override
public PreparedQuery> query() {
QueryExecutor, RowSetImpl, RowSet> builder = new QueryExecutor<>(RowSetImpl.FACTORY, RowSetImpl.COLLECTOR);
return new PreparedStatementQuery<>(builder);
}
void withPreparedStatement(PrepareOptions options, Tuple args, Handler> handler) {
if (context == Vertx.currentContext()) {
if (future == null) {
// Lazy statement;
PrepareStatementCommand prepare = new PrepareStatementCommand(sql, options, true, args.types());
conn.schedule(context, prepare).onComplete(promise);
future = promise.future();
}
future.onComplete(handler);
} else {
context.runOnContext(v -> withPreparedStatement(options, args, handler));
}
}
> void execute(Tuple args,
int fetch,
String cursorId,
boolean suspended,
QueryExecutor builder,
PromiseInternal p) {
withPreparedStatement(options, args, ar -> {
if (ar.succeeded()) {
builder.executeExtendedQuery(
conn,
ar.result(),
options,
autoCommit,
args,
fetch,
cursorId,
suspended,
p);
} else {
p.fail(ar.cause());
}
});
}
> void executeBatch(List argsList,
QueryExecutor builder,
PromiseInternal p) {
withPreparedStatement(options, argsList.get(0), ar -> {
if (ar.succeeded()) {
builder.executeBatchQuery(conn, options, ar.result(), autoCommit, argsList, p);
} else {
p.fail(ar.cause());
}
});
}
@Override
public Cursor cursor(Tuple args) {
return cursor((TupleInternal) args);
}
private Cursor cursor(TupleInternal args) {
return new CursorImpl(this, conn, context, autoCommit, args);
}
@Override
public Future close() {
if (closed.compareAndSet(false, true)) {
Promise promise = context.promise();
if (this.promise == null) {
CloseStatementCommand cmd = new CloseStatementCommand(future.result());
conn.schedule(context, cmd).onComplete(promise);
} else {
if (future == null) {
future = this.promise.future();
this.promise.fail("Closed");
}
future.onComplete(ar -> {
if (ar.succeeded()) {
CloseStatementCommand cmd = new CloseStatementCommand(ar.result());
conn.schedule(context, cmd).onComplete(promise);
} else {
promise.complete();
}
});
}
return promise.future();
} else {
return context.failedFuture("Already closed");
}
}
@Override
public RowStream createStream(int fetch, Tuple args) {
return new RowStreamImpl(this, context, fetch, args);
}
@Override
public void close(Handler> completionHandler) {
Future fut = close();
if (completionHandler != null) {
fut.onComplete(completionHandler);
}
}
void closeCursor(String cursorId, Promise promise) {
future.onComplete(ar -> {
if (ar.succeeded()) {
CloseCursorCommand cmd = new CloseCursorCommand(cursorId, ar.result());
conn.schedule(context, cmd).onComplete(promise);
} else {
promise.fail(ar.cause());
}
});
}
private class PreparedStatementQuery> extends QueryBase implements PreparedQuery {
public PreparedStatementQuery(QueryExecutor builder) {
super(builder);
}
@Override
protected > QueryBase copy(QueryExecutor builder) {
return new PreparedStatementQuery<>(builder);
}
@Override
public PreparedQuery> collecting(Collector collector) {
return (PreparedQuery>) super.collecting(collector);
}
@Override
public PreparedQuery> mapping(Function mapper) {
return (PreparedQuery>) super.mapping(mapper);
}
@Override
public void execute(Handler> handler) {
execute(ArrayTuple.EMPTY, handler);
}
@Override
public Future execute() {
return execute(ArrayTuple.EMPTY);
}
@Override
public void execute(Tuple args, Handler> handler) {
execute(args, context.promise(handler));
}
@Override
public Future execute(Tuple args) {
Promise promise = context.promise();
execute(args, promise);
return promise.future();
}
private void execute(Tuple args, PromiseInternal promise) {
PreparedStatementImpl.this.execute(args, 0, null, false, builder, promise);
}
public void executeBatch(List argsList, Handler> handler) {
executeBatch(argsList, context.promise(handler));
}
@Override
public Future executeBatch(List argsList) {
Promise promise = context.promise();
executeBatch(argsList, promise);
return promise.future();
}
private void executeBatch(List argsList, PromiseInternal promise) {
if (argsList.isEmpty()) {
promise.fail("Empty batch");
} else {
PreparedStatementImpl.this.executeBatch(argsList, builder, promise);
}
}
}
}