io.vertx.ext.mongo.impl.MongoIterableStream Maven / Gradle / Ivy
package io.vertx.ext.mongo.impl;
import com.mongodb.async.AsyncBatchCursor;
import com.mongodb.async.SingleResultCallback;
import com.mongodb.async.client.MongoIterable;
import io.vertx.core.Context;
import io.vertx.core.Handler;
import io.vertx.core.json.JsonObject;
import io.vertx.core.streams.ReadStream;
import io.vertx.core.streams.impl.InboundBuffer;
import java.util.List;
import java.util.concurrent.atomic.AtomicReference;
/**
* @author Thomas Segismont
*/
class MongoIterableStream implements ReadStream {
private final Context context;
private final MongoIterable mongoIterable;
private final int batchSize;
// All the following fields are guarded by this instance
private AsyncBatchCursor batchCursor;
private InboundBuffer queue;
private Handler exceptionHandler;
private Handler endHandler;
private boolean closed;
MongoIterableStream(Context context, MongoIterable mongoIterable, int batchSize) {
this.context = context;
this.mongoIterable = mongoIterable;
this.batchSize = batchSize;
this.queue = new InboundBuffer<>(context);
queue.drainHandler(v -> doRead());
queue.emptyHandler(v -> {
synchronized (this) {
if (closed) {
if (endHandler != null) {
endHandler.handle(null);
}
}
}
});
}
@Override
public synchronized MongoIterableStream exceptionHandler(Handler handler) {
this.exceptionHandler = handler;
return this;
}
@Override
public synchronized MongoIterableStream handler(Handler handler) {
queue.handler(handler);
if (handler == null) {
close();
} else {
SingleResultCallback> callback = (result, t) -> {
context.runOnContext(v -> {
synchronized (this) {
if (t != null) {
close();
handleException(t);
} else {
batchCursor = result;
batchCursor.setBatchSize(batchSize);
if (!closed) {
doRead();
}
}
}
});
};
try {
mongoIterable.batchCursor(callback);
} catch (Exception e) {
close();
handleException(e);
}
}
return this;
}
@Override
public MongoIterableStream pause() {
queue.pause();
return this;
}
@Override
public MongoIterableStream resume() {
queue.resume();
return this;
}
@Override
public ReadStream fetch(long amount) {
queue.fetch(amount);
return this;
}
// Always called from a synchronized method or block
private synchronized void doRead() {
context.>executeBlocking(fut -> {
batchCursor.next((result, t) -> {
if (t != null) {
fut.fail(t);
} else {
fut.complete(result);
}
});
}, true, ar -> {
synchronized (this) {
if (ar.succeeded()) {
List list = ar.result();
if (list != null) {
if (queue.write(list)) {
doRead();
}
} else {
close();
if (queue.isEmpty()) {
if (endHandler != null) {
endHandler.handle(null);
}
}
}
} else {
close();
handleException(ar.cause());
}
}
});
}
// Always called from a synchronized method or block
private void handleException(Throwable cause) {
if (exceptionHandler != null) {
exceptionHandler.handle(cause);
}
}
@Override
public synchronized MongoIterableStream endHandler(Handler handler) {
endHandler = handler;
return this;
}
// Always called from a synchronized method or block
private void close() {
if (closed) {
return;
}
closed = true;
AtomicReference cursorRef = new AtomicReference<>();
context.executeBlocking(fut -> {
synchronized (this) {
cursorRef.set(batchCursor);
}
AsyncBatchCursor cursor = cursorRef.get();
if (cursor != null) {
cursor.close();
}
fut.complete();
}, false, null);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy