|
16 | 16 | package com.mongodb.reactivestreams.client.internal;
|
17 | 17 |
|
18 | 18 | import com.mongodb.MongoClientSettings;
|
| 19 | +import com.mongodb.MongoCursorNotFoundException; |
19 | 20 | import com.mongodb.client.model.changestream.ChangeStreamDocument;
|
20 | 21 | import com.mongodb.client.result.InsertOneResult;
|
21 | 22 | import com.mongodb.event.CommandEvent;
|
| 23 | +import com.mongodb.event.CommandStartedEvent; |
22 | 24 | import com.mongodb.internal.connection.TestCommandListener;
|
23 | 25 | import com.mongodb.reactivestreams.client.FindPublisher;
|
24 | 26 | import com.mongodb.reactivestreams.client.MongoClient;
|
|
35 | 37 | import org.junit.jupiter.api.AfterEach;
|
36 | 38 | import org.junit.jupiter.api.BeforeEach;
|
37 | 39 | import org.junit.jupiter.api.DisplayName;
|
| 40 | +import org.junit.jupiter.api.Test; |
38 | 41 | import org.junit.jupiter.api.extension.ExtendWith;
|
39 | 42 | import org.mockito.Mock;
|
40 | 43 | import org.mockito.junit.jupiter.MockitoExtension;
|
41 | 44 | import org.reactivestreams.Publisher;
|
| 45 | +import reactor.core.publisher.Flux; |
| 46 | +import reactor.core.publisher.Hooks; |
42 | 47 | import reactor.core.publisher.Mono;
|
43 | 48 |
|
44 | 49 | import java.util.List;
|
| 50 | +import java.util.concurrent.atomic.AtomicBoolean; |
45 | 51 | import java.util.stream.Collectors;
|
46 | 52 | import java.util.stream.IntStream;
|
47 | 53 |
|
|
56 | 62 | import static java.util.Collections.singletonList;
|
57 | 63 | import static org.junit.jupiter.api.Assertions.assertAll;
|
58 | 64 | import static org.junit.jupiter.api.Assertions.assertEquals;
|
| 65 | +import static org.junit.jupiter.api.Assertions.assertFalse; |
59 | 66 | import static org.junit.jupiter.api.Assertions.assertIterableEquals;
|
60 | 67 | import static org.junit.jupiter.api.Assertions.assertNotNull;
|
61 | 68 | import static org.junit.jupiter.api.Assumptions.assumeTrue;
|
62 | 69 | import static org.mockito.Mockito.when;
|
63 | 70 |
|
64 |
| -import org.junit.jupiter.api.Test; |
65 |
| - |
66 | 71 | @ExtendWith(MockitoExtension.class)
|
67 | 72 | public class BatchCursorFluxTest {
|
68 | 73 |
|
@@ -299,6 +304,64 @@ void changeStreamPublisherCompletesAfterDroppingCollection() {
|
299 | 304 | subscriber.assertNoErrors();
|
300 | 305 | }
|
301 | 306 |
|
| 307 | + @Test |
| 308 | + @DisplayName("Ensure BatchCursor does not drop an error") |
| 309 | + public void testBatchCursorDoesNotDropAnError() { |
| 310 | + try { |
| 311 | + AtomicBoolean errorDropped = new AtomicBoolean(); |
| 312 | + Hooks.onErrorDropped(t -> errorDropped.set(true)); |
| 313 | + Mono.from(collection.insertMany(createDocs(200))).block(); |
| 314 | + |
| 315 | + Flux.fromStream(IntStream.range(1, 200).boxed()) |
| 316 | + .flatMap(i -> |
| 317 | + Flux.fromIterable(asList(1, 2)) |
| 318 | + .flatMap(x -> Flux.from(collection.find())) |
| 319 | + .take(1) |
| 320 | + |
| 321 | + ) |
| 322 | + .collectList() |
| 323 | + .block(TIMEOUT_DURATION); |
| 324 | + |
| 325 | + assertFalse(errorDropped.get()); |
| 326 | + } finally { |
| 327 | + Hooks.resetOnErrorDropped(); |
| 328 | + } |
| 329 | + } |
| 330 | + |
| 331 | + @Test |
| 332 | + @DisplayName("Ensure BatchCursor reports cursor errors") |
| 333 | + @SuppressWarnings("OptionalGetWithoutIsPresent") |
| 334 | + public void testBatchCursorReportsCursorErrors() { |
| 335 | + List<Document> docs = createDocs(200); |
| 336 | + Mono.from(collection.insertMany(docs)).block(TIMEOUT_DURATION); |
| 337 | + |
| 338 | + TestSubscriber<Document> subscriber = new TestSubscriber<>(); |
| 339 | + FindPublisher<Document> findPublisher = collection.find().batchSize(50); |
| 340 | + findPublisher.subscribe(subscriber); |
| 341 | + assertCommandNames(emptyList()); |
| 342 | + |
| 343 | + subscriber.requestMore(100); |
| 344 | + subscriber.assertReceivedOnNext(docs.subList(0, 100)); |
| 345 | + assertCommandNames(asList("find", "getMore")); |
| 346 | + |
| 347 | + BsonDocument getMoreCommand = commandListener.getCommandStartedEvents().stream() |
| 348 | + .filter(e -> e.getCommandName().equals("getMore")) |
| 349 | + .map(e -> ((CommandStartedEvent) e).getCommand()) |
| 350 | + .findFirst() |
| 351 | + .get(); |
| 352 | + |
| 353 | + Mono.from(client.getDatabase(getDefaultDatabaseName()).runCommand( |
| 354 | + new BsonDocument("killCursors", new BsonString(collection.getNamespace().getCollectionName())) |
| 355 | + .append("cursors", new BsonArray(singletonList(getMoreCommand.getNumber("getMore")))) |
| 356 | + )).block(TIMEOUT_DURATION); |
| 357 | + |
| 358 | + subscriber.requestMore(200); |
| 359 | + List<Throwable> onErrorEvents = subscriber.getOnErrorEvents(); |
| 360 | + subscriber.assertTerminalEvent(); |
| 361 | + assertEquals(1, onErrorEvents.size()); |
| 362 | + assertEquals(MongoCursorNotFoundException.class, onErrorEvents.get(0).getClass()); |
| 363 | + } |
| 364 | + |
302 | 365 | private void assertCommandNames(final List<String> commandNames) {
|
303 | 366 | assertIterableEquals(commandNames,
|
304 | 367 | commandListener.getCommandStartedEvents().stream().map(CommandEvent::getCommandName).collect(Collectors.toList()));
|
|
0 commit comments