forked from petabridge/akkadotnet-code-samples
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathProductTotalsActor.cs
96 lines (77 loc) · 2.72 KB
/
ProductTotalsActor.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
using Akka.Actor;
using Akka.Cluster.Sharding;
using Akka.Event;
using Akka.Persistence;
using CqrsSqlServer.Shared;
using CqrsSqlServer.Shared.Commands;
using CqrsSqlServer.Shared.Events;
using CqrsSqlServer.Shared.Queries;
namespace CqrsSqlServer.Backend.Actors;
/// <summary>
/// Manages the state for a given product.
/// </summary>
public sealed class ProductTotalsActor : ReceivePersistentActor
{
public static Props GetProps(string persistenceId)
{
return Props.Create(() => new ProductTotalsActor(persistenceId));
}
/// <summary>
/// Used to help differentiate what type of entity this is inside Akka.Persistence's database
/// </summary>
public const string TotalsEntityNameConstant = "totals";
private readonly ILoggingAdapter _log = Context.GetLogger();
public ProductTotalsActor(string persistenceId)
{
PersistenceId = $"{TotalsEntityNameConstant}-" + persistenceId;
State = new ProductState();
Recover<SnapshotOffer>(offer =>
{
if (offer.Snapshot is ProductState state)
{
State = state;
}
});
Recover<IProductEvent>(productEvent => { State = State.ProcessEvent(productEvent); });
Command<IProductCommand>(cmd =>
{
var response = State.ProcessCommand(cmd);
var sentResponse = false;
if (response.ResponseEvents.Count != 0)
{
PersistAll(response.ResponseEvents, productEvent =>
{
_log.Info("Processed: {0}", productEvent);
if (productEvent is ProductInventoryWarningEvent warning)
{
_log.Warning(warning.ToString());
}
State = State.ProcessEvent(productEvent);
if (!sentResponse) // otherwise we'll generate a response-per-event
{
sentResponse = true;
Sender.Tell(response);
}
if (LastSequenceNr % 10 == 0)
SaveSnapshot(State);
});
}
else
{
Sender.Tell(response);
}
});
Command<SaveSnapshotSuccess>(success => { });
Command<FetchProduct>(fetch =>
{
Sender.Tell(new FetchResult(State));
if (State.IsEmpty)
{
// we don't exist, so don't let `remember-entities` keep us alive
Context.Parent.Tell(new Passivate(PoisonPill.Instance));
}
});
}
public override string PersistenceId { get; }
public ProductState State { get; set; }
}