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

http://github.com/SpringSource/spring-data-mongodb · Java · 298 lines · 94 code · 34 blank · 170 comment · 3 complexity · c2723f350de969d6a7b2f3524df2cbb7 MD5 · raw file

  1. /*
  2. * Copyright 2018-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.aggregation;
  17. import java.util.Arrays;
  18. import java.util.Collection;
  19. import org.bson.Document;
  20. import org.springframework.util.Assert;
  21. /**
  22. * Gateway for
  23. * <a href="https://docs.mongodb.com/manual/meta/aggregation-quick-reference/#object-expression-operators">object
  24. * expression operators</a>.
  25. *
  26. * @author Christoph Strobl
  27. * @since 2.1
  28. */
  29. public class ObjectOperators {
  30. /**
  31. * Take the value referenced by given {@literal fieldReference}.
  32. *
  33. * @param fieldReference must not be {@literal null}.
  34. * @return new instance of {@link ObjectOperatorFactory}.
  35. */
  36. public static ObjectOperatorFactory valueOf(String fieldReference) {
  37. return new ObjectOperatorFactory(Fields.field(fieldReference));
  38. }
  39. /**
  40. * Take the value provided by the given {@link AggregationExpression}.
  41. *
  42. * @param expression must not be {@literal null}.
  43. * @return new instance of {@link ObjectOperatorFactory}.
  44. */
  45. public static ObjectOperatorFactory valueOf(AggregationExpression expression) {
  46. return new ObjectOperatorFactory(expression);
  47. }
  48. /**
  49. * @author Christoph Strobl
  50. */
  51. public static class ObjectOperatorFactory {
  52. private final Object value;
  53. /**
  54. * Creates new {@link ObjectOperatorFactory} for given {@literal value}.
  55. *
  56. * @param value must not be {@literal null}.
  57. */
  58. public ObjectOperatorFactory(Object value) {
  59. Assert.notNull(value, "Value must not be null!");
  60. this.value = value;
  61. }
  62. /**
  63. * Creates new {@link MergeObjects aggregation expression} that takes the associated value and uses
  64. * {@literal $mergeObjects} as an accumulator within the {@literal $group} stage. <br />
  65. * <strong>NOTE:</strong> Requires MongoDB 4.0 or later.
  66. *
  67. * @return new instance of {@link MergeObjects}.
  68. */
  69. public MergeObjects merge() {
  70. return MergeObjects.merge(value);
  71. }
  72. /**
  73. * Creates new {@link MergeObjects aggregation expression} that takes the associated value and combines it with the
  74. * given values (documents or mapped objects) into a single document. <br />
  75. * <strong>NOTE:</strong> Requires MongoDB 4.0 or later.
  76. *
  77. * @return new instance of {@link MergeObjects}.
  78. */
  79. public MergeObjects mergeWith(Object... values) {
  80. return merge().mergeWith(values);
  81. }
  82. /**
  83. * Creates new {@link MergeObjects aggregation expression} that takes the associated value and combines it with the
  84. * values of the given {@link Field field references} into a single document. <br />
  85. * <strong>NOTE:</strong> Requires MongoDB 4.0 or later.
  86. *
  87. * @return new instance of {@link MergeObjects}.
  88. */
  89. public MergeObjects mergeWithValuesOf(String... fieldReferences) {
  90. return merge().mergeWithValuesOf(fieldReferences);
  91. }
  92. /**
  93. * Creates new {@link MergeObjects aggregation expression} that takes the associated value and combines it with the
  94. * result values of the given {@link Aggregation expressions} into a single document. <br />
  95. * <strong>NOTE:</strong> Requires MongoDB 4.0 or later.
  96. *
  97. * @return new instance of {@link MergeObjects}.
  98. */
  99. public MergeObjects mergeWithValuesOf(AggregationExpression... expression) {
  100. return merge().mergeWithValuesOf(expression);
  101. }
  102. /**
  103. * Creates new {@link ObjectToArray aggregation expression} that takes the associated value and converts it to an
  104. * array of {@link Document documents} that contain two fields {@literal k} and {@literal v} each. <br />
  105. * <strong>NOTE:</strong> Requires MongoDB 3.6 or later.
  106. *
  107. * @since 2.1
  108. */
  109. public ObjectToArray toArray() {
  110. return ObjectToArray.toArray(value);
  111. }
  112. }
  113. /**
  114. * {@link AggregationExpression} for {@code $mergeObjects} that combines multiple documents into a single document.
  115. * <br />
  116. * <strong>NOTE:</strong> Requires MongoDB 4.0 or later.
  117. *
  118. * @author Christoph Strobl
  119. * @see <a href=
  120. * "https://docs.mongodb.com/manual/reference/operator/aggregation/mergeObjects/">https://docs.mongodb.com/manual/reference/operator/aggregation/mergeObjects/</a>
  121. * @since 2.1
  122. */
  123. public static class MergeObjects extends AbstractAggregationExpression {
  124. private MergeObjects(Object value) {
  125. super(value);
  126. }
  127. /**
  128. * Creates new {@link MergeObjects aggregation expression} that takes given values and combines them into a single
  129. * document. <br />
  130. *
  131. * @param values must not be {@literal null}.
  132. * @return new instance of {@link MergeObjects}.
  133. */
  134. public static MergeObjects merge(Object... values) {
  135. return new MergeObjects(Arrays.asList(values));
  136. }
  137. /**
  138. * Creates new {@link MergeObjects aggregation expression} that takes the given {@link Field field references} and
  139. * combines them into a single document.
  140. *
  141. * @param fieldReferences must not be {@literal null}.
  142. * @return new instance of {@link MergeObjects}.
  143. */
  144. public static MergeObjects mergeValuesOf(String... fieldReferences) {
  145. return merge(Arrays.stream(fieldReferences).map(Fields::field).toArray());
  146. }
  147. /**
  148. * Creates new {@link MergeObjects aggregation expression} that takes the result of the given {@link Aggregation
  149. * expressions} and combines them into a single document.
  150. *
  151. * @param expressions must not be {@literal null}.
  152. * @return new instance of {@link MergeObjects}.
  153. */
  154. public static MergeObjects mergeValuesOf(AggregationExpression... expressions) {
  155. return merge(expressions);
  156. }
  157. /**
  158. * Creates new {@link MergeObjects aggregation expression} by adding the given {@link Field field references}.
  159. *
  160. * @param fieldReferences must not be {@literal null}.
  161. * @return new instance of {@link MergeObjects}.
  162. */
  163. public MergeObjects mergeWithValuesOf(String... fieldReferences) {
  164. return mergeWith(Arrays.stream(fieldReferences).map(Fields::field).toArray());
  165. }
  166. /**
  167. * Creates new {@link MergeObjects aggregation expression} by adding the given {@link AggregationExpression
  168. * expressions}.
  169. *
  170. * @param expression must not be {@literal null}.
  171. * @return new instance of {@link MergeObjects}.
  172. */
  173. public MergeObjects mergeWithValuesOf(AggregationExpression... expression) {
  174. return mergeWith(expression);
  175. }
  176. /**
  177. * Creates new {@link MergeObjects aggregation expression} by adding the given values (documents or mapped objects).
  178. *
  179. * @param values must not be {@literal null}.
  180. * @return new instance of {@link MergeObjects}.
  181. */
  182. public MergeObjects mergeWith(Object... values) {
  183. return new MergeObjects(append(Arrays.asList(values)));
  184. }
  185. /*
  186. * (non-Javadoc)
  187. * @see org.springframework.data.mongodb.core.aggregation.AbstractAggregationExpression#toDocument(java.lang.Object, org.springframework.data.mongodb.core.aggregation.AggregationOperationContext)
  188. */
  189. @Override
  190. public Document toDocument(Object value, AggregationOperationContext context) {
  191. return super.toDocument(potentiallyExtractSingleValue(value), context);
  192. }
  193. @SuppressWarnings("unchecked")
  194. private Object potentiallyExtractSingleValue(Object value) {
  195. if (value instanceof Collection) {
  196. Collection<Object> collection = ((Collection<Object>) value);
  197. if (collection.size() == 1) {
  198. return collection.iterator().next();
  199. }
  200. }
  201. return value;
  202. }
  203. /*
  204. * (non-Javadoc)
  205. * @see org.springframework.data.mongodb.core.aggregation.AbstractAggregationExpression#getMongoMethod()
  206. */
  207. @Override
  208. protected String getMongoMethod() {
  209. return "$mergeObjects";
  210. }
  211. }
  212. /**
  213. * {@link AggregationExpression} for {@code $objectToArray} that converts a document to an array of {@link Document
  214. * documents} that each contains two fields {@literal k} and {@literal v}. <br />
  215. * <strong>NOTE:</strong> Requires MongoDB 3.6 or later.
  216. *
  217. * @author Christoph Strobl
  218. * @see <a href=
  219. * "https://docs.mongodb.com/manual/reference/operator/aggregation/objectToArray/">https://docs.mongodb.com/manual/reference/operator/aggregation/objectToArray/</a>
  220. * @since 2.1
  221. */
  222. public static class ObjectToArray extends AbstractAggregationExpression {
  223. private ObjectToArray(Object value) {
  224. super(value);
  225. }
  226. /**
  227. * Creates new {@link ObjectToArray aggregation expression} that takes the value pointed to by given {@link Field
  228. * fieldReference} and converts it to an array.
  229. *
  230. * @param fieldReference must not be {@literal null}.
  231. * @return new instance of {@link ObjectToArray}.
  232. */
  233. public static ObjectToArray valueOfToArray(String fieldReference) {
  234. return toArray(Fields.field(fieldReference));
  235. }
  236. /**
  237. * Creates new {@link ObjectToArray aggregation expression} that takes the result value of the given
  238. * {@link AggregationExpression expression} and converts it to an array.
  239. *
  240. * @param expression must not be {@literal null}.
  241. * @return new instance of {@link ObjectToArray}.
  242. */
  243. public static ObjectToArray valueOfToArray(AggregationExpression expression) {
  244. return toArray(expression);
  245. }
  246. /**
  247. * Creates new {@link ObjectToArray aggregation expression} that takes the given value and converts it to an array.
  248. *
  249. * @param value must not be {@literal null}.
  250. * @return new instance of {@link ObjectToArray}.
  251. */
  252. public static ObjectToArray toArray(Object value) {
  253. return new ObjectToArray(value);
  254. }
  255. /*
  256. * (non-Javadoc)
  257. * @see org.springframework.data.mongodb.core.aggregation.AbstractAggregationExpression#getMongoMethod()
  258. */
  259. @Override
  260. protected String getMongoMethod() {
  261. return "$objectToArray";
  262. }
  263. }
  264. }