/modelling/src/main/java/org/axonframework/modelling/command/AggregateLifecycle.java

http://github.com/AxonFramework/AxonFramework · Java · 217 lines · 56 code · 20 blank · 141 comment · 1 complexity · a843df5d15ad3e75898a7d808c8ce3c6 MD5 · raw file

  1. /*
  2. * Copyright (c) 2010-2018. Axon Framework
  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 org.axonframework.modelling.command;
  17. import org.axonframework.messaging.MetaData;
  18. import org.axonframework.messaging.Scope;
  19. import org.axonframework.messaging.ScopeDescriptor;
  20. import java.util.concurrent.Callable;
  21. /**
  22. * Abstract base class of a component that models an aggregate's life cycle.
  23. */
  24. public abstract class AggregateLifecycle extends Scope {
  25. /**
  26. * Apply a {@link org.axonframework.eventhandling.DomainEventMessage} with given payload and metadata (metadata
  27. * from interceptors will be combined with the provided metadata). Applying events means they are immediately
  28. * applied (published) to the aggregate and scheduled for publication to other event handlers.
  29. * <p/>
  30. * The event is applied on all entities part of this aggregate. If the event is applied from an event handler of the
  31. * aggregate and additional events need to be applied that depends on state changes brought about by the first event
  32. * use the returned {@link ApplyMore} instance.
  33. *
  34. * @param payload the payload of the event to apply
  35. * @param metaData any meta-data that must be registered with the Event
  36. * @return a gizmo to apply additional events after the given event has been processed by the entire aggregate
  37. * @see ApplyMore
  38. */
  39. public static ApplyMore apply(Object payload, MetaData metaData) {
  40. return AggregateLifecycle.getInstance().doApply(payload, metaData);
  41. }
  42. /**
  43. * Apply a {@link org.axonframework.eventhandling.DomainEventMessage} with given payload without metadata (though
  44. * interceptors can also be used to provide metadata). Applying events means they are immediately applied
  45. * (published) to the aggregate and scheduled for publication to other event handlers.
  46. * <p/>
  47. * The event is applied on all entities part of this aggregate. If the event is applied from an event handler of the
  48. * aggregate and additional events need to be applied that depends on state changes brought about by the first event
  49. * use the returned {@link ApplyMore} instance.
  50. *
  51. * @param payload the payload of the event to apply
  52. * @return a gizmo to apply additional events after the given event has been processed by the entire aggregate
  53. * @see ApplyMore
  54. */
  55. public static ApplyMore apply(Object payload) {
  56. return AggregateLifecycle.getInstance().doApply(payload, MetaData.emptyInstance());
  57. }
  58. /**
  59. * Creates a new aggregate instance. In order for new aggregate to be created, a {@link Repository} should be
  60. * available to the current aggregate. {@link Repository} of an aggregate to be created is exposed to the current
  61. * aggregate via {@link RepositoryProvider}.
  62. *
  63. * @param <T> type of new aggregate to be created
  64. * @param aggregateType type of new aggregate to be created
  65. * @param factoryMethod factory method which creates new aggregate
  66. * @return a new aggregate instance
  67. * @throws Exception thrown if something goes wrong during instantiation of new aggregate
  68. */
  69. public static <T> Aggregate<T> createNew(Class<T> aggregateType, Callable<T> factoryMethod)
  70. throws Exception {
  71. if (!isLive()) {
  72. throw new UnsupportedOperationException(
  73. "Aggregate is still initializing its state, creation of new aggregates is not possible");
  74. }
  75. return getInstance().doCreateNew(aggregateType, factoryMethod);
  76. }
  77. /**
  78. * Indicates whether this Aggregate instance is 'live'. Events applied to a 'live' Aggregate represent events that
  79. * are currently happening, as opposed to events representing historic decisions used to reconstruct the
  80. * Aggregate's state.
  81. *
  82. * @return {@code true} if the aggregate is 'live', {@code false} if the aggregate is initializing state based on
  83. * historic events
  84. */
  85. public static boolean isLive() {
  86. return AggregateLifecycle.getInstance().getIsLive();
  87. }
  88. /**
  89. * Gets the version of the aggregate.
  90. *
  91. * @return the current version of the aggregate
  92. */
  93. public static Long getVersion() {
  94. return getInstance().version();
  95. }
  96. /**
  97. * Marks this aggregate as deleted, instructing a repository to remove that aggregate at an appropriate time.
  98. * <p/>
  99. * Note that different repository implementations may react differently to aggregates marked for deletion.
  100. * Typically, Event Sourced Repositories will ignore the marking and expect deletion to be provided as part of Event
  101. * information.
  102. */
  103. public static void markDeleted() {
  104. getInstance().doMarkDeleted();
  105. }
  106. /**
  107. * Get the current {@link AggregateLifecycle} instance for the current thread. If none exists an {@link
  108. * IllegalStateException} is thrown.
  109. *
  110. * @return the thread's current {@link AggregateLifecycle}
  111. */
  112. protected static AggregateLifecycle getInstance() {
  113. return Scope.getCurrentScope();
  114. }
  115. /**
  116. * Indicates whether this Aggregate instance is 'live'. This means events currently applied represent events that
  117. * are currently happening, as opposed to events representing historic decisions.
  118. *
  119. * @return {@code true} if the aggregate is 'live', {@code false} if the aggregate is initializing state based on
  120. * historic events
  121. */
  122. protected abstract boolean getIsLive();
  123. /**
  124. * Gets the version of the aggregate.
  125. *
  126. * @return the current version of the aggregate
  127. */
  128. protected abstract Long version();
  129. /**
  130. * Marks this aggregate as deleted. Implementations may react differently to aggregates marked for deletion.
  131. * Typically, Event Sourced Repositories will ignore the marking and expect deletion to be provided as part of Event
  132. * information.
  133. */
  134. protected abstract void doMarkDeleted();
  135. /**
  136. * Apply a {@link org.axonframework.eventhandling.DomainEventMessage} with given payload and metadata (metadata from
  137. * interceptors will be combined with the provided metadata). The event should be applied to the aggregate
  138. * immediately and scheduled for publication to other event handlers.
  139. * <p/>
  140. * The event should be applied on all entities part of this aggregate. If the event is applied from an event handler
  141. * of the aggregate and additional events need to be applied that depends on state changes brought about by the
  142. * first event the returned {@link ApplyMore} instance should allow for additional events to be applied after this
  143. * event.
  144. *
  145. * @param payload the payload of the event to apply
  146. * @param metaData any meta-data that must be registered with the Event
  147. * @return a gizmo to apply additional events after the given event has been processed by the entire aggregate
  148. * @see ApplyMore
  149. */
  150. protected abstract <T> ApplyMore doApply(T payload, MetaData metaData);
  151. /**
  152. * Creates a new aggregate instance. In order for new aggregate to be created, a {@link Repository} should be
  153. * available to the current aggregate. {@link Repository} of an aggregate to be created is exposed to the current
  154. * aggregate via {@link RepositoryProvider}.
  155. *
  156. * @param <T> type of new aggregate to be created
  157. * @param aggregateType type of new aggregate to be created
  158. * @param factoryMethod factory method which creates new aggregate
  159. * @return a new aggregate instance
  160. * @throws Exception thrown if something goes wrong during instantiation of new aggregate
  161. */
  162. protected abstract <T> Aggregate<T> doCreateNew(Class<T> aggregateType, Callable<T> factoryMethod) throws Exception;
  163. /**
  164. * Executes the given task. While the task is being executed the current aggregate will be registered with the
  165. * current thread as the 'current' aggregate.
  166. *
  167. * @param task the task to execute on the aggregate
  168. */
  169. protected void execute(Runnable task) {
  170. try {
  171. executeWithResult(() -> {
  172. task.run();
  173. return null;
  174. });
  175. } catch (RuntimeException e) {
  176. throw e;
  177. } catch (Exception e) {
  178. throw new AggregateInvocationException("Exception while invoking a task for an aggregate", e);
  179. }
  180. }
  181. @Override
  182. public ScopeDescriptor describeScope() {
  183. return new AggregateScopeDescriptor(type(), this::identifier);
  184. }
  185. /**
  186. * Retrieve a {@link String} denoting the type of this Aggregate.
  187. *
  188. * @return a {@link String} denoting the type of this Aggregate
  189. */
  190. protected abstract String type();
  191. /**
  192. * Retrieve a {@link Object} denoting the identifier of this Aggregate.
  193. *
  194. * @return a {@link Object} denoting the identifier of this Aggregate
  195. */
  196. protected abstract Object identifier();
  197. }