Documentation Coverage Report
Current view: top level - Akka.Net/Cqrs.Akka/Domain - AkkaAggregateRoot.cs Hit Total Coverage
Version: 4.0 Artefacts: 8 8 100.0 %
Date: 2019-11-24 03:15:41

          Line data    Source code
       1             : #region Copyright
       2             : // // -----------------------------------------------------------------------
       3             : // // <copyright company="Chinchilla Software Limited">
       4             : // //   Copyright Chinchilla Software Limited. All rights reserved.
       5             : // // </copyright>
       6             : // // -----------------------------------------------------------------------
       7             : #endregion
       8             : 
       9             : using System;
      10             : using System.Collections.Generic;
      11             : using System.Collections.ObjectModel;
      12             : using System.Linq;
      13             : using Akka.Actor;
      14             : using Chinchilla.Logging;
      15             : using Cqrs.Akka.Snapshots;
      16             : using Cqrs.Authentication;
      17             : using Cqrs.Commands;
      18             : using Cqrs.Domain;
      19             : using Cqrs.Domain.Exceptions;
      20             : using Cqrs.Events;
      21             : using Cqrs.Infrastructure;
      22             : 
      23             : namespace Cqrs.Akka.Domain
      24             : {
      25             :         /// <summary>
      26             :         /// An <see cref="IAggregateRoot{TAuthenticationToken}"/> that is safe to use within Akka.NET
      27             :         /// </summary>
      28             :         /// <typeparam name="TAuthenticationToken">The <see cref="Type"/> of authentication token.</typeparam>
      29             :         public abstract class AkkaAggregateRoot<TAuthenticationToken>
      30             :                 : ReceiveActor // PersistentActor 
      31             :                 , IAggregateRoot<TAuthenticationToken>
      32           1 :         {
      33             :                 /// <summary>
      34             :                 /// Gets or sets the <see cref="IUnitOfWork{TAuthenticationToken}"/>.
      35             :                 /// </summary>
      36             :                 protected IUnitOfWork<TAuthenticationToken> UnitOfWork { get; set; }
      37             : 
      38             :                 /// <summary>
      39             :                 /// Gets or sets the <see cref="IAkkaAggregateRepository{TAuthenticationToken}"/>.
      40             :                 /// </summary>
      41             :                 protected IAkkaAggregateRepository<TAuthenticationToken> Repository { get; set; }
      42             : 
      43             :                 /// <summary>
      44             :                 /// Gets or sets the <see cref="ILogger"/>.
      45             :                 /// </summary>
      46             :                 protected ILogger Logger { get; set; }
      47             : 
      48             :                 /// <summary>
      49             :                 /// Gets or sets the <see cref="ICorrelationIdHelper"/>.
      50             :                 /// </summary>
      51             :                 protected ICorrelationIdHelper CorrelationIdHelper { get; set; }
      52             : 
      53             :                 /// <summary>
      54             :                 /// Gets or sets the <see cref="IAuthenticationTokenHelper{TAuthenticationToken}"/>.
      55             :                 /// </summary>
      56             :                 protected IAuthenticationTokenHelper<TAuthenticationToken> AuthenticationTokenHelper { get; set; }
      57             : 
      58             :                 private ICollection<IEvent<TAuthenticationToken>> Changes { get; set; }
      59             : 
      60             :                 /// <summary>
      61             :                 /// The identifier of this <see cref="IAggregateRoot{TAuthenticationToken}"/>.
      62             :                 /// </summary>
      63             :                 public Guid Id { get; protected set; }
      64             : 
      65             :                 /// <summary>
      66             :                 /// The current version of this <see cref="IAggregateRoot{TAuthenticationToken}"/>.
      67             :                 /// </summary>
      68             :                 public int Version { get; protected set; }
      69             : 
      70             :                 /// <summary>
      71             :                 /// Instantiates a new instance of <see cref="AkkaAggregateRoot{TAuthenticationToken}"/>
      72             :                 /// </summary>
      73           1 :                 protected AkkaAggregateRoot(IUnitOfWork<TAuthenticationToken> unitOfWork, ILogger logger, IAkkaAggregateRepository<TAuthenticationToken> repository, ICorrelationIdHelper correlationIdHelper, IAuthenticationTokenHelper<TAuthenticationToken> authenticationTokenHelper)
      74             :                 {
      75             :                         UnitOfWork = unitOfWork;
      76             :                         Logger = logger;
      77             :                         Repository = repository;
      78             :                         CorrelationIdHelper = correlationIdHelper;
      79             :                         AuthenticationTokenHelper = authenticationTokenHelper;
      80             :                         Changes = new ReadOnlyCollection<IEvent<TAuthenticationToken>>(new List<IEvent<TAuthenticationToken>>());
      81             :                 }
      82             : 
      83             :                 #region Overrides of ActorBase
      84             : 
      85             :                 /// <summary>
      86             :                 /// User overridable callback.
      87             :                 ///                 <p/>
      88             :                 ///                 Is called when an Actor is started.
      89             :                 ///                 Actors are automatically started asynchronously when created.
      90             :                 ///                 Empty default implementation.
      91             :                 /// </summary>
      92           1 :                 protected override void PreStart()
      93             :                 {
      94             :                         base.PreStart();
      95             :                         Repository.LoadAggregateHistory(this, throwExceptionOnNoEvents: false);
      96             :                 }
      97             : 
      98             :                 #endregion
      99             : 
     100             :                 /// <summary>
     101             :                 /// Executes the provided <paramref name="action"/> passing it the provided <paramref name="command"/>,
     102             :                 /// then calls <see cref="AggregateRepository{TAuthenticationToken}.PublishEvent"/>
     103             :                 /// </summary>
     104           1 :                 protected virtual void Execute<TCommand>(Action<TCommand> action, TCommand command)
     105             :                         where TCommand : ICommand<TAuthenticationToken>
     106             :                 {
     107             :                         Type baseType = GetType().BaseType;
     108             :                         UnitOfWork.Add(this, baseType != null && baseType.Name == typeof(AkkaSnapshotAggregateRoot<,>).Name);
     109             :                         try
     110             :                         {
     111             :                                 AuthenticationTokenHelper.SetAuthenticationToken(command.AuthenticationToken);
     112             :                                 CorrelationIdHelper.SetCorrelationId(command.CorrelationId);
     113             :                                 action(command);
     114             : 
     115             :                                 UnitOfWork.Commit();
     116             : 
     117             :                                 Sender.Tell(true, Self);
     118             :                         }
     119             :                         catch(Exception exception)
     120             :                         {
     121             :                                 Logger.LogError("Executing an Akka.net request failed.", exception: exception, metaData: new Dictionary<string, object> { { "Type", GetType() }, { "Command", command } });
     122             :                                 Sender.Tell(false, Self);
     123             :                                 throw;
     124             :                         }
     125             :                 }
     126             : 
     127             :                 /// <summary>
     128             :                 /// Get all applied changes that haven't yet been committed.
     129             :                 /// </summary>
     130           1 :                 public IEnumerable<IEvent<TAuthenticationToken>> GetUncommittedChanges()
     131             :                 {
     132             :                         return Changes;
     133             :                 }
     134             : 
     135             :                 /// <summary>
     136             :                 /// Mark all applied changes as committed, increment <see cref="Version"/> and flush the internal collection of changes.
     137             :                 /// </summary>
     138           1 :                 public virtual void MarkChangesAsCommitted()
     139             :                 {
     140             :                         Version = Version + Changes.Count;
     141             :                         Changes = new ReadOnlyCollection<IEvent<TAuthenticationToken>>(new List<IEvent<TAuthenticationToken>>());
     142             :                 }
     143             : 
     144             :                 /// <summary>
     145             :                 /// Apply all the <see cref="IEvent{TAuthenticationToken}">events</see> in <paramref name="history"/>
     146             :                 /// using event replay to this instance.
     147             :                 /// </summary>
     148           1 :                 public virtual void LoadFromHistory(IEnumerable<IEvent<TAuthenticationToken>> history)
     149             :                 {
     150             :                         Type aggregateType = GetType();
     151             :                         foreach (IEvent<TAuthenticationToken> @event in history.OrderBy(e =>e.Version))
     152             :                         {
     153             :                                 if (@event.Version != Version + 1)
     154             :                                         throw new EventsOutOfOrderException(@event.GetIdentity(), aggregateType, Version + 1, @event.Version);
     155             :                                 ApplyChange(@event, true);
     156             :                         }
     157             :                 }
     158             : 
     159             :                 /// <summary>
     160             :                 /// Call the "Apply" method with a signature matching the provided <paramref name="event"/> without using event replay to this instance.
     161             :                 /// </summary>
     162             :                 /// <remarks>
     163             :                 /// This means a method named "Apply", with return type void and one parameter must exist to be applied.
     164             :                 /// If no method exists, nothing is applied
     165             :                 /// The parameter type must match exactly the <see cref="Type"/> of the provided <paramref name="event"/>.
     166             :                 /// </remarks>
     167           1 :                 protected virtual void ApplyChange(IEvent<TAuthenticationToken> @event)
     168             :                 {
     169             :                         ApplyChange(@event, false);
     170             :                 }
     171             : 
     172             :                 private void ApplyChange(IEvent<TAuthenticationToken> @event, bool isEventReplay)
     173             :                 {
     174             :                         this.AsDynamic().Apply(@event);
     175             :                         if (!isEventReplay)
     176             :                         {
     177             :                                 Changes = new ReadOnlyCollection<IEvent<TAuthenticationToken>>(new []{@event}.Concat(Changes).ToList());
     178             :                         }
     179             :                         else
     180             :                         {
     181             :                                 Id = @event.GetIdentity();
     182             :                                 Version++;
     183             :                         }
     184             :                 }
     185             :         }
     186             : }

Generated by: LCOV version 1.13