/spring-data-mongodb/src/main/java/org/springframework/data/mongodb/core/CountQuery.java

http://github.com/SpringSource/spring-data-mongodb · Java · 227 lines · 132 code · 53 blank · 42 comment · 31 complexity · 0034c79ba81104deb1adac433812f2cc MD5 · raw file

  1. /*
  2. * Copyright 2019-2021 the original author or authors.
  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. * https://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 org.springframework.data.mongodb.core;
  17. import java.util.ArrayList;
  18. import java.util.Arrays;
  19. import java.util.Collection;
  20. import java.util.Collections;
  21. import java.util.List;
  22. import java.util.Map;
  23. import org.bson.Document;
  24. import org.springframework.data.geo.Point;
  25. import org.springframework.lang.Nullable;
  26. import org.springframework.util.ObjectUtils;
  27. /**
  28. * Value object representing a count query. Count queries using {@code $near} or {@code $nearSphere} require a rewrite
  29. * to {@code $geoWithin}.
  30. *
  31. * @author Christoph Strobl
  32. * @author Mark Paluch
  33. * @since 3.0
  34. */
  35. class CountQuery {
  36. private Document source;
  37. private CountQuery(Document source) {
  38. this.source = source;
  39. }
  40. public static CountQuery of(Document source) {
  41. return new CountQuery(source);
  42. }
  43. /**
  44. * Returns the query {@link Document} that can be used with {@code countDocuments()}. Potentially rewrites the query
  45. * to be usable with {@code countDocuments()}.
  46. *
  47. * @return the query {@link Document} that can be used with {@code countDocuments()}.
  48. */
  49. public Document toQueryDocument() {
  50. if (!requiresRewrite(source)) {
  51. return source;
  52. }
  53. Document target = new Document();
  54. for (Map.Entry<String, Object> entry : source.entrySet()) {
  55. if (entry.getValue() instanceof Document && requiresRewrite(entry.getValue())) {
  56. Document theValue = (Document) entry.getValue();
  57. target.putAll(createGeoWithin(entry.getKey(), theValue, source.get("$and")));
  58. continue;
  59. }
  60. if (entry.getValue() instanceof Collection && requiresRewrite(entry.getValue())) {
  61. Collection<?> source = (Collection<?>) entry.getValue();
  62. target.put(entry.getKey(), rewriteCollection(source));
  63. continue;
  64. }
  65. if ("$and".equals(entry.getKey()) && target.containsKey("$and")) {
  66. // Expect $and to be processed with Document and createGeoWithin.
  67. continue;
  68. }
  69. target.put(entry.getKey(), entry.getValue());
  70. }
  71. return target;
  72. }
  73. /**
  74. * @param valueToInspect
  75. * @return {@code true} if the enclosing element needs to be rewritten.
  76. */
  77. private boolean requiresRewrite(Object valueToInspect) {
  78. if (valueToInspect instanceof Document) {
  79. return requiresRewrite((Document) valueToInspect);
  80. }
  81. if (valueToInspect instanceof Collection) {
  82. return requiresRewrite((Collection) valueToInspect);
  83. }
  84. return false;
  85. }
  86. private boolean requiresRewrite(Collection<?> collection) {
  87. for (Object o : collection) {
  88. if (o instanceof Document && requiresRewrite((Document) o)) {
  89. return true;
  90. }
  91. }
  92. return false;
  93. }
  94. private boolean requiresRewrite(Document document) {
  95. if (containsNear(document)) {
  96. return true;
  97. }
  98. for (Object entry : document.values()) {
  99. if (requiresRewrite(entry)) {
  100. return true;
  101. }
  102. }
  103. return false;
  104. }
  105. private Collection<Object> rewriteCollection(Collection<?> source) {
  106. Collection<Object> rewrittenCollection = new ArrayList<>(source.size());
  107. for (Object item : source) {
  108. if (item instanceof Document && requiresRewrite(item)) {
  109. rewrittenCollection.add(CountQuery.of((Document) item).toQueryDocument());
  110. } else {
  111. rewrittenCollection.add(item);
  112. }
  113. }
  114. return rewrittenCollection;
  115. }
  116. /**
  117. * Rewrite the near query for field {@code key} to {@code $geoWithin}.
  118. *
  119. * @param key the queried field.
  120. * @param source source {@link Document}.
  121. * @param $and potentially existing {@code $and} condition.
  122. * @return the rewritten query {@link Document}.
  123. */
  124. private static Document createGeoWithin(String key, Document source, @Nullable Object $and) {
  125. boolean spheric = source.containsKey("$nearSphere");
  126. Object $near = spheric ? source.get("$nearSphere") : source.get("$near");
  127. Number maxDistance = source.containsKey("$maxDistance") ? (Number) source.get("$maxDistance") : Double.MAX_VALUE;
  128. List<Object> $centerMax = Arrays.asList(toCenterCoordinates($near), maxDistance);
  129. Document $geoWithinMax = new Document("$geoWithin",
  130. new Document(spheric ? "$centerSphere" : "$center", $centerMax));
  131. if (!containsNearWithMinDistance(source)) {
  132. return new Document(key, $geoWithinMax);
  133. }
  134. Number minDistance = (Number) source.get("$minDistance");
  135. List<Object> $centerMin = Arrays.asList(toCenterCoordinates($near), minDistance);
  136. Document $geoWithinMin = new Document("$geoWithin",
  137. new Document(spheric ? "$centerSphere" : "$center", $centerMin));
  138. List<Document> criteria = new ArrayList<>();
  139. if ($and != null) {
  140. if ($and instanceof Collection) {
  141. criteria.addAll((Collection) $and);
  142. } else {
  143. throw new IllegalArgumentException(
  144. "Cannot rewrite query as it contains an '$and' element that is not a Collection!: Offending element: "
  145. + $and);
  146. }
  147. }
  148. criteria.add(new Document("$nor", Collections.singletonList(new Document(key, $geoWithinMin))));
  149. criteria.add(new Document(key, $geoWithinMax));
  150. return new Document("$and", criteria);
  151. }
  152. private static boolean containsNear(Document source) {
  153. return source.containsKey("$near") || source.containsKey("$nearSphere");
  154. }
  155. private static boolean containsNearWithMinDistance(Document source) {
  156. if (!containsNear(source)) {
  157. return false;
  158. }
  159. return source.containsKey("$minDistance");
  160. }
  161. private static Object toCenterCoordinates(Object value) {
  162. if (ObjectUtils.isArray(value)) {
  163. return value;
  164. }
  165. if (value instanceof Point) {
  166. return Arrays.asList(((Point) value).getX(), ((Point) value).getY());
  167. }
  168. if (value instanceof Document && ((Document) value).containsKey("x")) {
  169. Document point = (Document) value;
  170. return Arrays.asList(point.get("x"), point.get("y"));
  171. }
  172. return value;
  173. }
  174. }