/driver/src/main/com/mongodb/FindIterableImpl.java

http://github.com/mongodb/mongo-java-driver · Java · 215 lines · 170 code · 30 blank · 15 comment · 1 complexity · e42384e8d096b6168597497369aed67b MD5 · raw file

  1. /*
  2. * Copyright (c) 2008-2014 MongoDB, Inc.
  3. *
  4. * Licensed under the Apache License, Version 2.0 (the "License");
  5. * you may not use this file except in compliance with the License.
  6. * You may obtain a copy of the License at
  7. *
  8. * http://www.apache.org/licenses/LICENSE-2.0
  9. *
  10. * Unless required by applicable law or agreed to in writing, software
  11. * distributed under the License is distributed on an "AS IS" BASIS,
  12. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. * See the License for the specific language governing permissions and
  14. * limitations under the License.
  15. */
  16. package com.mongodb;
  17. import com.mongodb.client.FindIterable;
  18. import com.mongodb.client.MongoCursor;
  19. import com.mongodb.client.MongoIterable;
  20. import com.mongodb.client.model.Collation;
  21. import com.mongodb.client.model.FindOptions;
  22. import com.mongodb.operation.BatchCursor;
  23. import com.mongodb.operation.FindOperation;
  24. import com.mongodb.operation.OperationExecutor;
  25. import org.bson.BsonDocument;
  26. import org.bson.codecs.configuration.CodecRegistry;
  27. import org.bson.conversions.Bson;
  28. import java.util.Collection;
  29. import java.util.concurrent.TimeUnit;
  30. import static com.mongodb.assertions.Assertions.notNull;
  31. import static java.util.concurrent.TimeUnit.MILLISECONDS;
  32. final class FindIterableImpl<TDocument, TResult> implements FindIterable<TResult> {
  33. private final MongoNamespace namespace;
  34. private final Class<TDocument> documentClass;
  35. private final Class<TResult> resultClass;
  36. private final ReadPreference readPreference;
  37. private final ReadConcern readConcern;
  38. private final CodecRegistry codecRegistry;
  39. private final OperationExecutor executor;
  40. private final FindOptions findOptions;
  41. private final Collation collation;
  42. private Bson filter;
  43. FindIterableImpl(final MongoNamespace namespace, final Class<TDocument> documentClass, final Class<TResult> resultClass,
  44. final CodecRegistry codecRegistry, final ReadPreference readPreference, final ReadConcern readConcern,
  45. final OperationExecutor executor, final Bson filter, final FindOptions findOptions, final Collation collation) {
  46. this.namespace = notNull("namespace", namespace);
  47. this.documentClass = notNull("documentClass", documentClass);
  48. this.resultClass = notNull("resultClass", resultClass);
  49. this.codecRegistry = notNull("codecRegistry", codecRegistry);
  50. this.readPreference = notNull("readPreference", readPreference);
  51. this.readConcern = notNull("readConcern", readConcern);
  52. this.executor = notNull("executor", executor);
  53. this.filter = notNull("filter", filter);
  54. this.findOptions = notNull("findOptions", findOptions);
  55. this.collation = collation;
  56. }
  57. @Override
  58. public FindIterable<TResult> filter(final Bson filter) {
  59. this.filter = filter;
  60. return this;
  61. }
  62. @Override
  63. public FindIterable<TResult> limit(final int limit) {
  64. findOptions.limit(limit);
  65. return this;
  66. }
  67. @Override
  68. public FindIterable<TResult> skip(final int skip) {
  69. findOptions.skip(skip);
  70. return this;
  71. }
  72. @Override
  73. public FindIterable<TResult> maxTime(final long maxTime, final TimeUnit timeUnit) {
  74. notNull("timeUnit", timeUnit);
  75. findOptions.maxTime(maxTime, timeUnit);
  76. return this;
  77. }
  78. @Override
  79. public FindIterable<TResult> maxAwaitTime(final long maxAwaitTime, final TimeUnit timeUnit) {
  80. notNull("timeUnit", timeUnit);
  81. findOptions.maxAwaitTime(maxAwaitTime, timeUnit);
  82. return this;
  83. }
  84. @Override
  85. public FindIterable<TResult> batchSize(final int batchSize) {
  86. findOptions.batchSize(batchSize);
  87. return this;
  88. }
  89. @Override
  90. public FindIterable<TResult> modifiers(final Bson modifiers) {
  91. findOptions.modifiers(modifiers);
  92. return this;
  93. }
  94. @Override
  95. public FindIterable<TResult> projection(final Bson projection) {
  96. findOptions.projection(projection);
  97. return this;
  98. }
  99. @Override
  100. public FindIterable<TResult> sort(final Bson sort) {
  101. findOptions.sort(sort);
  102. return this;
  103. }
  104. @Override
  105. public FindIterable<TResult> noCursorTimeout(final boolean noCursorTimeout) {
  106. findOptions.noCursorTimeout(noCursorTimeout);
  107. return this;
  108. }
  109. @Override
  110. public FindIterable<TResult> oplogReplay(final boolean oplogReplay) {
  111. findOptions.oplogReplay(oplogReplay);
  112. return this;
  113. }
  114. @Override
  115. public FindIterable<TResult> partial(final boolean partial) {
  116. findOptions.partial(partial);
  117. return this;
  118. }
  119. @Override
  120. public FindIterable<TResult> cursorType(final CursorType cursorType) {
  121. findOptions.cursorType(cursorType);
  122. return this;
  123. }
  124. @Override
  125. public MongoCursor<TResult> iterator() {
  126. return execute().iterator();
  127. }
  128. @Override
  129. public TResult first() {
  130. return execute().first();
  131. }
  132. @Override
  133. public <U> MongoIterable<U> map(final Function<TResult, U> mapper) {
  134. return new MappingIterable<TResult, U>(this, mapper);
  135. }
  136. @Override
  137. public void forEach(final Block<? super TResult> block) {
  138. execute().forEach(block);
  139. }
  140. @Override
  141. public <A extends Collection<? super TResult>> A into(final A target) {
  142. return execute().into(target);
  143. }
  144. private MongoIterable<TResult> execute() {
  145. return new FindOperationIterable(createQueryOperation(), readPreference, executor);
  146. }
  147. private FindOperation<TResult> createQueryOperation() {
  148. return new FindOperation<TResult>(namespace, codecRegistry.get(resultClass))
  149. .filter(filter.toBsonDocument(documentClass, codecRegistry))
  150. .batchSize(findOptions.getBatchSize())
  151. .skip(findOptions.getSkip())
  152. .limit(findOptions.getLimit())
  153. .maxTime(findOptions.getMaxTime(MILLISECONDS), MILLISECONDS)
  154. .maxAwaitTime(findOptions.getMaxAwaitTime(MILLISECONDS), MILLISECONDS)
  155. .modifiers(toBsonDocument(findOptions.getModifiers()))
  156. .projection(toBsonDocument(findOptions.getProjection()))
  157. .sort(toBsonDocument(findOptions.getSort()))
  158. .cursorType(findOptions.getCursorType())
  159. .noCursorTimeout(findOptions.isNoCursorTimeout())
  160. .oplogReplay(findOptions.isOplogReplay())
  161. .partial(findOptions.isPartial())
  162. .slaveOk(readPreference.isSlaveOk())
  163. .readConcern(readConcern)
  164. .collation(collation);
  165. }
  166. private BsonDocument toBsonDocument(final Bson document) {
  167. return document == null ? null : document.toBsonDocument(documentClass, codecRegistry);
  168. }
  169. private final class FindOperationIterable extends OperationIterable<TResult> {
  170. private final ReadPreference readPreference;
  171. private final OperationExecutor executor;
  172. FindOperationIterable(final FindOperation<TResult> operation, final ReadPreference readPreference,
  173. final OperationExecutor executor) {
  174. super(operation, readPreference, executor);
  175. this.readPreference = readPreference;
  176. this.executor = executor;
  177. }
  178. @Override
  179. public TResult first() {
  180. FindOperation<TResult> findFirstOperation = createQueryOperation().batchSize(0).limit(-1);
  181. BatchCursor<TResult> batchCursor = executor.execute(findFirstOperation, readPreference);
  182. return batchCursor.hasNext() ? batchCursor.next().iterator().next() : null;
  183. }
  184. }
  185. }