package io.vertx.sqlclient.impl;
import io.vertx.core.AsyncResult;
import io.vertx.core.Future;
import io.vertx.core.Handler;
import io.vertx.core.Promise;
import io.vertx.core.impl.ContextInternal;
import io.vertx.core.impl.future.PromiseInternal;
import io.vertx.core.spi.metrics.ClientMetrics;
import io.vertx.sqlclient.Cursor;
import io.vertx.sqlclient.Row;
import io.vertx.sqlclient.RowSet;
import io.vertx.sqlclient.impl.tracing.QueryTracer;
import java.util.UUID;
public class CursorImpl implements Cursor {
private final Connection conn;
private final QueryTracer tracer;
private final ClientMetrics metrics;
private final PreparedStatementImpl ps;
private final ContextInternal context;
private final boolean autoCommit;
private final TupleInternal params;
private String id;
private boolean closed;
private QueryResultBuilder<RowSet<Row>, RowSetImpl<Row>, RowSet<Row>> result;
CursorImpl(PreparedStatementImpl ps, Connection conn, QueryTracer tracer, ClientMetrics metrics, ContextInternal context, boolean autoCommit, TupleInternal params) {
this.ps = ps;
this.conn = conn;
this.tracer = tracer;
this.metrics = metrics;
this.context = context;
this.autoCommit = autoCommit;
this.params = params;
}
@Override
public synchronized boolean hasMore() {
if (result == null) {
throw new IllegalStateException("No current cursor read");
}
return result.isSuspended();
}
@Override
public void read(int count, Handler<AsyncResult<RowSet<Row>>> handler) {
Future<RowSet<Row>> fut = read(count);
if (handler != null) {
fut.onComplete(handler);
}
}
@Override
public synchronized Future<RowSet<Row>> read(int count) {
PromiseInternal<RowSet<Row>> promise = context.promise();
ps.withPreparedStatement(params, ar -> {
if (ar.succeeded()) {
PreparedStatement preparedStatement = ar.result();
QueryExecutor<RowSet<Row>, RowSetImpl<Row>, RowSet<Row>> builder = new QueryExecutor<>(tracer, metrics, RowSetImpl.FACTORY, RowSetImpl.COLLECTOR);
if (id == null) {
id = UUID.randomUUID().toString();
this.result = builder.executeExtendedQuery(conn, preparedStatement, autoCommit, params, count, id, false, promise);
} else if (this.result.isSuspended()) {
this.result = builder.executeExtendedQuery(conn, preparedStatement, autoCommit, params, count, id, true, promise);
} else {
throw new IllegalStateException();
}
} else {
promise.fail(ar.cause());
}
});
return promise.future();
}
@Override
public synchronized boolean isClosed() {
return closed;
}
@Override
public synchronized void close(Handler<AsyncResult<Void>> completionHandler) {
close (context.promise(completionHandler));
}
@Override
public synchronized Future<Void> close() {
Promise<Void> promise = context.promise();
close (promise);
return promise.future();
}
private synchronized void close(Promise<Void> promise) {
if (!closed) {
closed = true;
if (id == null) {
promise.complete();
} else {
String id = this.id;
this.id = null;
result = null;
ps.closeCursor(id, promise);
}
}
}
}