Cleipnir.Flows.MassTransit
3.0.3-alpha
Prefix Reserved
dotnet add package Cleipnir.Flows.MassTransit --version 3.0.3-alpha
NuGet\Install-Package Cleipnir.Flows.MassTransit -Version 3.0.3-alpha
<PackageReference Include="Cleipnir.Flows.MassTransit" Version="3.0.3-alpha" />
paket add Cleipnir.Flows.MassTransit --version 3.0.3-alpha
#r "nuget: Cleipnir.Flows.MassTransit, 3.0.3-alpha"
// Install Cleipnir.Flows.MassTransit as a Cake Addin #addin nuget:?package=Cleipnir.Flows.MassTransit&version=3.0.3-alpha&prerelease // Install Cleipnir.Flows.MassTransit as a Cake Tool #tool nuget:?package=Cleipnir.Flows.MassTransit&version=3.0.3-alpha&prerelease
<p align="center"> <img src="./Docs/cleipnir.png" alt="logo" /> <br> Simply making fault-tolerant code simple <br> </p>
Cleipnir Flows
Cleipnir Flows is a simple and intuitive workflow-as-code .NET framework - ensuring your code completes despite: failures, restarts, deployments, versioning etc.
It is similar to Azure Durable Functions - but simpler, less restrictive and tailored for ASP.NET / generic host services.
It works for both RPC and Message-based communication.
Introduction video
Getting Started
To get started simply perform the following three steps in an ASP.NET service:
Firstly, install the Cleipnir.Flows nuget package (using either Postgres, SqlServer or MySQL as persistence layer). I.e.
Install-Package Cleipnir.Flows.Postgres
Secondly, add the following to the setup in Program.cs
:
builder.Services.AddFlows(c => c
.UsePostgresSqlStore(connectionString)
.RegisterFlowsAutomatically()
);
Finally, implement your flow:
public class OrderFlow : Flow<Order>
{
private readonly IPaymentProviderClient _paymentProviderClient;
private readonly IEmailClient _emailClient;
private readonly ILogisticsClient _logisticsClient;
public OrderFlow(IPaymentProviderClient paymentProviderClient, IEmailClient emailClient, ILogisticsClient logisticsClient)
{
_paymentProviderClient = paymentProviderClient;
_emailClient = emailClient;
_logisticsClient = logisticsClient;
}
public override async Task Run(Order order)
{
await _paymentProviderClient.Reserve(order.CustomerId, order.TransactionId, order.TotalPrice);
await _logisticsClient.ShipProducts(order.CustomerId, order.ProductIds);
await _paymentProviderClient.Capture(order.TransactionId);
await _emailClient.SendOrderConfirmation(order.CustomerId, order.ProductIds);
}
}
The flow can then be started using the corresponding source generated Flows-type:
[ApiController]
[Route("[controller]")]
public class OrderController : ControllerBase
{
private readonly OrderFlows _orderFlows;
public OrderController(OrderFlows orderFlows) => _orderFlows = orderFlows;
[HttpPost]
public async Task Post(Order order) => await _orderFlows.Run(order.OrderId, order);
}
Congrats, any non-completed Order flows are now automatically restarted by the framework.
However, the real benefit of the framework comes from:
- Simplifying how code will be executed after a crash/restart - using the AtMostOnce and AtLeastOnce helper-methods
- Awaiting external messages declaratively using Reactive Programming
- Simple testability & versioning
Examples
As an example is worth a thousand lines of documentation - various useful examples are presented in the following section:
1: Avoid executing already completed code again if a flow is restarted:
public class AtLeastOnceFlow : Flow<string, string>
{
private readonly PuzzleSolverService _puzzleSolverService = new();
public override async Task<string> Run(string hashCode)
{
var solution = await Effect.Capture(
id: "PuzzleSolution",
work: () => _puzzleSolverService.SolveCryptographicPuzzle(hashCode)
);
return solution;
}
}
2: Ensure a flow step is executed at-most-once:
public class AtMostOnceFlow : Flow<string>
{
private readonly RocketSender _rocketSender = new();
public override async Task Run(string rocketId)
{
await Effect.Capture(
id: "FireRocket",
_rocketSender.FireRocket,
ResiliencyLevel.AtMostOnce
);
}
}
3: Wait for 2 external messages before continuing flow (source code):
public class WaitForMessagesFlow : Flow<string>
{
public override async Task Run(string param)
{
await Messages
.OfTypes<FundsReserved, InventoryLocked>()
.Take(2)
.Completion(maxWait: TimeSpan.FromSeconds(30));
System.Console.WriteLine("Complete order-processing");
}
}
When the max wait duration has passed the flow is automatically suspended in order to save resources. Thus, the flow can also be suspended immediately when all messages have not been received:
await Messages
.OfTypes<FundsReserved, InventoryLocked>()
.Take(2)
.Completion();
4: Emit a signal to a flow (source code):
var messagesWriter = flows.MessagesWriter(orderId);
await messagesWriter.AppendMessage(new FundsReserved(orderId), idempotencyKey: nameof(FundsReserved));
5: Restart a failed flow (source code):
var controlPanel = await flows.ControlPanel(flowId);
controlPanel!.Param = "valid parameter";
await controlPanel.ReInvoke();
6: Postpone a running flow (without taking in-memory resources) (source code):
public class PostponeFlow : Flow<string>
{
private readonly ExternalService _externalService = new();
public override async Task Run(string orderId)
{
if (await _externalService.IsOverloaded())
Postpone(delay: TimeSpan.FromMinutes(10));
//execute rest of the flow
}
}
7: Add metrics middleware (source code):
public class MetricsMiddleware : IMiddleware
{
private Action IncrementCompletedFlowsCounter { get; }
private Action IncrementFailedFlowsCounter { get; }
private Action IncrementRestartedFlowsCounter { get; }
public MetricsMiddleware(Action incrementCompletedFlowsCounter, Action incrementFailedFlowsCounter, Action incrementRestartedFlowsCounter)
{
IncrementCompletedFlowsCounter = incrementCompletedFlowsCounter;
IncrementFailedFlowsCounter = incrementFailedFlowsCounter;
IncrementRestartedFlowsCounter = incrementRestartedFlowsCounter;
}
public async Task<Result<TResult>> Run<TFlow, TParam, TResult>(
TParam param,
Context context,
Next<TFlow, TParam, TResult> next) where TParam : notnull
{
var started = workflow.Effect.TryGet<bool>(id: "Started", out _);
if (started)
IncrementRestartedFlowsCounter();
else
await workflow.Effect.Upsert("Started", true);
var result = await next(param, workflow);
if (result.Outcome == Outcome.Fail)
IncrementFailedFlowsCounter();
else if (result.Outcome == Outcome.Succeed)
IncrementCompletedFlowsCounter();
return result;
}
}
What is it about?
When distributed systems needs to cooperator in order to fulfill some business process a system crash or restart may leave the system in an inconsistent state.
Consider the following order-flow:
public async Task ProcessOrder(Order order)
{
await _paymentProviderClient.Reserve(order.TransactionId, order.CustomerId, order.TotalPrice);
await _logisticsClient.ShipProducts(order.CustomerId, order.ProductIds);
await _paymentProviderClient.Capture(order.TransactionId);
await _emailClient.SendOrderConfirmation(order.CustomerId, order.ProductIds);
}
Currently, the flow is not resilient against crashes or restarts.
For instance, if the process crashes just before capturing the funds from the payment provider then the ordered products are shipped to the customer but nothing is deducted from the customer’s credit card. Not an ideal situation for the business. No matter how we rearrange the flow a crash might lead to either situation:
- products are shipped to the customer without payment being deducted from the customer’s credit card
- payment is deducted from the customer’s credit card but products are never shipped
Ensuring flow-restart on crashes or restarts:
Thus, to rectify the situation we must ensure that the flow is restarted if it did not complete in a previous execution.
RPC-solution
Consider the following Order-flow:
public class OrderFlow : Flow<Order>
{
private readonly IPaymentProviderClient _paymentProviderClient;
private readonly IEmailClient _emailClient;
private readonly ILogisticsClient _logisticsClient;
public OrderFlow(IPaymentProviderClient paymentProviderClient, IEmailClient emailClient, ILogisticsClient logisticsClient)
{
_paymentProviderClient = paymentProviderClient;
_emailClient = emailClient;
_logisticsClient = logisticsClient;
}
public async Task ProcessOrder(Order order)
{
Log.Logger.ForContext<OrderFlow>().Information($"ORDER_PROCESSOR: Processing of order '{order.OrderId}' started");
var transactionId = Guid.Empty;
await _paymentProviderClient.Reserve(order.CustomerId, transactionId, order.TotalPrice);
await _logisticsClient.ShipProducts(order.CustomerId, order.ProductIds);
await _paymentProviderClient.Capture(transactionId);
await _emailClient.SendOrderConfirmation(order.CustomerId, order.ProductIds);
Log.Logger.ForContext<OrderFlow>().Information($"ORDER_PROCESSOR: Processing of order '{order.OrderId}' completed");
}
}
Sometimes simply wrapping a business flow inside the framework is enough.
This would be the case if all the steps in the flow were idempotent. In that situation it is fine to call an endpoint multiple times without causing unintended side-effects.
At-least-once & Idempotency
However, in the order-flow presented here this is not the case.
The payment provider requires the caller to provide a transaction-id. Thus, the same transaction-id must be provided when re-executing the flow.
In Cleipnir this challenge is solved by wrapping non-determinism inside effects.
public async Task ProcessOrder(Order order)
{
Log.Logger.Information($"ORDER_PROCESSOR: Processing of order '{order.OrderId}' started");
var transactionId = Effect.Capture("TransactionId", Guid.NewGuid);
await _paymentProviderClient.Reserve(transactionId, order.CustomerId, order.TotalPrice);
await _logisticsClient.ShipProducts(order.CustomerId, order.ProductIds);
await _paymentProviderClient.Capture(transactionId);
await _emailClient.SendOrderConfirmation(order.CustomerId, order.ProductIds);
Log.Logger.ForContext<OrderProcessor>().Information($"Processing of order '{order.OrderId}' completed");
}
At-most-once API:
For the sake of presenting the framework’s versatility let us assume that the logistics’ API is not idempotent and that it is out of our control to change that.
Thus, every time a successful call is made to the logistics service the content of the order is shipped to the customer.
As a result the order-flow must fail if it is restarted and:
- a request was previously sent to logistics-service
- but no response was received.
This can again be accomplished by using effects:
public async Task ProcessOrder(Order order)
{
Log.Logger.Information($"ORDER_PROCESSOR: Processing of order '{order.OrderId}' started");
var transactionId = await Effect.Capture("TransactionId", Guid.NewGuid);
await _paymentProviderClient.Reserve(order.CustomerId, transactionId, order.TotalPrice);
await Effect.Capture(
id: "ShipProducts",
work: () => _logisticsClient.ShipProducts(order.CustomerId, order.ProductIds)
);
await _paymentProviderClient.Capture(transactionId);
await _emailClient.SendOrderConfirmation(order.CustomerId, order.ProductIds);
Log.Logger.ForContext<OrderProcessor>().Information($"Processing of order '{order.OrderId}' completed");
}
A failed/exception throwing flow is not automatically retried by the framework.
Instead, it must be manually restarted by using the flow's associated control-panel.
Control Panel:
Using the flow’s control panel both the parameter and scrapbook may be changed before the flow is retried.
For instance, assuming it is determined that the products where not shipped for a certain order, then the following code re-invokes the order with the state changed accordingly.
var controlPanel = await flows.ControlPanel(order.OrderId);
await controlPanel!.Effects.Remove("ShipProducts");
await controlPanel.ReInvoke();
Message-based Solution
Message- or event-driven system are omnipresent in enterprise architectures today.
They fundamentally differ from RPC-based in that:
- messages related to the same order are not delivered to the same process.
This has huge implications in how a flow must be implemented and as a result a simple sequential flow - as in the case of the order-flow:
- becomes fragmented and hard to reason about
- inefficient - each time a message is received the entire state must be reestablished
- inflexible
Cleipnir Flows takes a novel approach by piggy-backing on the features described so far and using event-sourcing and reactive programming together to form a simple and extremely useful abstraction.
As a result the order-flow can be implemented as follows:
public async Task ProcessOrder(Order order)
{
Log.Logger.Information($"ORDER_PROCESSOR: Processing of order '{order.OrderId}' started");
await _bus.Send(new ReserveFunds(order.OrderId, order.TotalPrice, Scrapbook.TransactionId, order.CustomerId));
await Messages.NextOfType<FundsReserved>();
await _bus.Send(new ShipProducts(order.OrderId, order.CustomerId, order.ProductIds));
await Messages.NextOfType<ProductsShipped>();
await _bus.Send(new CaptureFunds(order.OrderId, order.CustomerId, Scrapbook.TransactionId));
await Messages.NextOfType<FundsCaptured>();
await _bus.Send(new SendOrderConfirmationEmail(order.OrderId, order.CustomerId));
await Messages.NextOfType<OrderConfirmationEmailSent>();
Log.Logger.ForContext<OrderProcessor>().Information($"Processing of order '{order.OrderId}' completed");
}
There is a bit more going on in the example above compared to the previous RPC-example. However, the flow is actually very similar to RPC-based. It is sequential and robust. If the flow crashes and is restarted it will continue from the point it got to before the crash.
It is noted that the message broker in the example is just a stand-in - thus not a framework concept - for RabbitMQ, Kafka or some other messaging infrastructure client.
In a real application the message broker would be replaced with the actual way the application broadcasts a message/event to other services.
Furthermore, each flow has an associated private event source called Messages. When events are received from the network they can be placed into the relevant flow's event source - thereby allowing the flow to continue.
Did you know? |
---|
The framework allows awaiting events both in-memory or suspending the invocation until an event has been appended to the event source. </br>Thus, allowing the developer to find the sweet-spot per use-case between performance and releasing resources. |
Product | Versions Compatible and additional computed target framework versions. |
---|---|
.NET | net8.0 is compatible. net8.0-android was computed. net8.0-browser was computed. net8.0-ios was computed. net8.0-maccatalyst was computed. net8.0-macos was computed. net8.0-tvos was computed. net8.0-windows was computed. |
-
net8.0
- Cleipnir.Flows (>= 3.0.2)
- MassTransit (>= 8.2.3)
NuGet packages
This package is not used by any NuGet packages.
GitHub repositories
This package is not used by any popular GitHub repositories.
Version | Downloads | Last updated | |
---|---|---|---|
3.0.3-alpha | 78 | 7/25/2024 | |
3.0.2-alpha | 51 | 7/25/2024 | |
3.0.1-alpha | 56 | 7/25/2024 |