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

http://github.com/SpringSource/spring-data-mongodb · Java · 291 lines · 54 code · 31 blank · 206 comment · 0 complexity · 08c5ea8d8b1613b9f2d66a11aa80d1e4 MD5 · raw file

  1. /*
  2. * Copyright 2017-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.Optional;
  18. import org.springframework.data.mongodb.core.aggregation.AggregationUpdate;
  19. import org.springframework.data.mongodb.core.query.CriteriaDefinition;
  20. import org.springframework.data.mongodb.core.query.Query;
  21. import org.springframework.data.mongodb.core.query.Update;
  22. import org.springframework.data.mongodb.core.query.UpdateDefinition;
  23. import org.springframework.lang.Nullable;
  24. import com.mongodb.client.result.UpdateResult;
  25. /**
  26. * {@link ExecutableUpdateOperation} allows creation and execution of MongoDB update / findAndModify / findAndReplace
  27. * operations in a fluent API style. <br />
  28. * The starting {@literal domainType} is used for mapping the {@link Query} provided via {@code matching}, as well as
  29. * the {@link Update} via {@code apply} into the MongoDB specific representations. The collection to operate on is by
  30. * default derived from the initial {@literal domainType} and can be defined there via
  31. * {@link org.springframework.data.mongodb.core.mapping.Document}. Using {@code inCollection} allows to override the
  32. * collection name for the execution.
  33. *
  34. * <pre>
  35. * <code>
  36. * update(Jedi.class)
  37. * .inCollection("star-wars")
  38. * .matching(query(where("firstname").is("luke")))
  39. * .apply(new Update().set("lastname", "skywalker"))
  40. * .upsert();
  41. * </code>
  42. * </pre>
  43. *
  44. * @author Christoph Strobl
  45. * @author Mark Paluch
  46. * @since 2.0
  47. */
  48. public interface ExecutableUpdateOperation {
  49. /**
  50. * Start creating an update operation for the given {@literal domainType}.
  51. *
  52. * @param domainType must not be {@literal null}.
  53. * @return new instance of {@link ExecutableUpdate}.
  54. * @throws IllegalArgumentException if domainType is {@literal null}.
  55. */
  56. <T> ExecutableUpdate<T> update(Class<T> domainType);
  57. /**
  58. * Trigger findAndModify execution by calling one of the terminating methods.
  59. *
  60. * @author Christoph Strobl
  61. * @author Mark Paluch
  62. * @since 2.0
  63. */
  64. interface TerminatingFindAndModify<T> {
  65. /**
  66. * Find, modify and return the first matching document.
  67. *
  68. * @return {@link Optional#empty()} if nothing found.
  69. */
  70. default Optional<T> findAndModify() {
  71. return Optional.ofNullable(findAndModifyValue());
  72. }
  73. /**
  74. * Find, modify and return the first matching document.
  75. *
  76. * @return {@literal null} if nothing found.
  77. */
  78. @Nullable
  79. T findAndModifyValue();
  80. }
  81. /**
  82. * Trigger
  83. * <a href="https://docs.mongodb.com/manual/reference/method/db.collection.findOneAndReplace/">findOneAndReplace</a>
  84. * execution by calling one of the terminating methods.
  85. *
  86. * @author Mark Paluch
  87. * @since 2.1
  88. */
  89. interface TerminatingFindAndReplace<T> {
  90. /**
  91. * Find, replace and return the first matching document.
  92. *
  93. * @return {@link Optional#empty()} if nothing found.
  94. */
  95. default Optional<T> findAndReplace() {
  96. return Optional.ofNullable(findAndReplaceValue());
  97. }
  98. /**
  99. * Find, replace and return the first matching document.
  100. *
  101. * @return {@literal null} if nothing found.
  102. */
  103. @Nullable
  104. T findAndReplaceValue();
  105. }
  106. /**
  107. * Trigger update execution by calling one of the terminating methods.
  108. *
  109. * @author Christoph Strobl
  110. * @since 2.0
  111. */
  112. interface TerminatingUpdate<T> extends TerminatingFindAndModify<T>, FindAndModifyWithOptions<T> {
  113. /**
  114. * Update all matching documents in the collection.
  115. *
  116. * @return never {@literal null}.
  117. */
  118. UpdateResult all();
  119. /**
  120. * Update the first document in the collection.
  121. *
  122. * @return never {@literal null}.
  123. */
  124. UpdateResult first();
  125. /**
  126. * Creates a new document if no documents match the filter query or updates the matching ones.
  127. *
  128. * @return never {@literal null}.
  129. */
  130. UpdateResult upsert();
  131. }
  132. /**
  133. * Declare the {@link Update} to apply.
  134. *
  135. * @author Christoph Strobl
  136. * @since 2.0
  137. */
  138. interface UpdateWithUpdate<T> {
  139. /**
  140. * Set the {@link UpdateDefinition} to be applied.
  141. *
  142. * @param update must not be {@literal null}.
  143. * @return new instance of {@link TerminatingUpdate}.
  144. * @throws IllegalArgumentException if update is {@literal null}.
  145. * @since 3.0
  146. * @see Update
  147. * @see AggregationUpdate
  148. */
  149. TerminatingUpdate<T> apply(UpdateDefinition update);
  150. /**
  151. * Specify {@code replacement} object.
  152. *
  153. * @param replacement must not be {@literal null}.
  154. * @return new instance of {@link FindAndReplaceOptions}.
  155. * @throws IllegalArgumentException if options is {@literal null}.
  156. * @since 2.1
  157. */
  158. FindAndReplaceWithProjection<T> replaceWith(T replacement);
  159. }
  160. /**
  161. * Explicitly define the name of the collection to perform operation in.
  162. *
  163. * @author Christoph Strobl
  164. * @since 2.0
  165. */
  166. interface UpdateWithCollection<T> {
  167. /**
  168. * Explicitly set the name of the collection to perform the query on. <br />
  169. * Skip this step to use the default collection derived from the domain type.
  170. *
  171. * @param collection must not be {@literal null} nor {@literal empty}.
  172. * @return new instance of {@link UpdateWithCollection}.
  173. * @throws IllegalArgumentException if collection is {@literal null}.
  174. */
  175. UpdateWithQuery<T> inCollection(String collection);
  176. }
  177. /**
  178. * Define a filter query for the {@link Update}.
  179. *
  180. * @author Christoph Strobl
  181. * @since 2.0
  182. */
  183. interface UpdateWithQuery<T> extends UpdateWithUpdate<T> {
  184. /**
  185. * Filter documents by given {@literal query}.
  186. *
  187. * @param query must not be {@literal null}.
  188. * @return new instance of {@link UpdateWithQuery}.
  189. * @throws IllegalArgumentException if query is {@literal null}.
  190. */
  191. UpdateWithUpdate<T> matching(Query query);
  192. /**
  193. * Set the filter {@link CriteriaDefinition criteria} to be used.
  194. *
  195. * @param criteria must not be {@literal null}.
  196. * @return new instance of {@link UpdateWithUpdate}.
  197. * @throws IllegalArgumentException if query is {@literal null}.
  198. * @since 3.0
  199. */
  200. default UpdateWithUpdate<T> matching(CriteriaDefinition criteria) {
  201. return matching(Query.query(criteria));
  202. }
  203. }
  204. /**
  205. * Define {@link FindAndModifyOptions}.
  206. *
  207. * @author Christoph Strobl
  208. * @since 2.0
  209. */
  210. interface FindAndModifyWithOptions<T> {
  211. /**
  212. * Explicitly define {@link FindAndModifyOptions} for the {@link Update}.
  213. *
  214. * @param options must not be {@literal null}.
  215. * @return new instance of {@link FindAndModifyWithOptions}.
  216. * @throws IllegalArgumentException if options is {@literal null}.
  217. */
  218. TerminatingFindAndModify<T> withOptions(FindAndModifyOptions options);
  219. }
  220. /**
  221. * Define {@link FindAndReplaceOptions}.
  222. *
  223. * @author Mark Paluch
  224. * @author Christoph Strobl
  225. * @since 2.1
  226. */
  227. interface FindAndReplaceWithOptions<T> extends TerminatingFindAndReplace<T> {
  228. /**
  229. * Explicitly define {@link FindAndReplaceOptions} for the {@link Update}.
  230. *
  231. * @param options must not be {@literal null}.
  232. * @return new instance of {@link FindAndReplaceOptions}.
  233. * @throws IllegalArgumentException if options is {@literal null}.
  234. */
  235. FindAndReplaceWithProjection<T> withOptions(FindAndReplaceOptions options);
  236. }
  237. /**
  238. * Result type override (Optional).
  239. *
  240. * @author Christoph Strobl
  241. * @since 2.1
  242. */
  243. interface FindAndReplaceWithProjection<T> extends FindAndReplaceWithOptions<T> {
  244. /**
  245. * Define the target type fields should be mapped to. <br />
  246. * Skip this step if you are anyway only interested in the original domain type.
  247. *
  248. * @param resultType must not be {@literal null}.
  249. * @param <R> result type.
  250. * @return new instance of {@link FindAndReplaceWithProjection}.
  251. * @throws IllegalArgumentException if resultType is {@literal null}.
  252. */
  253. <R> FindAndReplaceWithOptions<R> as(Class<R> resultType);
  254. }
  255. /**
  256. * @author Christoph Strobl
  257. * @since 2.0
  258. */
  259. interface ExecutableUpdate<T> extends UpdateWithCollection<T>, UpdateWithQuery<T>, UpdateWithUpdate<T> {}
  260. }