Skip to content

Latest commit

 

History

History
229 lines (203 loc) · 8.49 KB

README.md

File metadata and controls

229 lines (203 loc) · 8.49 KB

Build Status codecov NuGet version (CoRProcessor)

CoRProcessor Framework

Overview 🌟

The CoRProcessor framework provides a way to implement the Chain of Responsibility (CoR) pattern in .NET applications. It allows you to define a chain of processors that can handle a request in sequence, with support for adding before, after, and finally actions, as well as exception handling.

Getting Started 🚀

Installation 📦

To use the CoRProcessor framework, simply add the CoRProcessor package to your project.

dotnet add package CoRProcessor

Define a Processor 🛠️

Processors must implement the IChainProcessor<T> interface. Here's an example of a simple processor:

public class SampleProcessor : IChainProcessor<MyData>
{
    public async Task<MyData> Handle(MyData data, CancellationToken token = default)
    {
        // Process the data
        Console.WriteLine("Processing in SampleProcessor");

        // Call the next processor in the chain
        return Task.FromResult(data);
    }
    
    public FuncDelegate<MyData> CompensateOnFailure { get; set; }
}

Compensation ↩️

The CoRProcessor framework also supports compensation. If a processor throws an exception, you can specify an action to be executed to compensate for the error.

public class SampleProcessor : IChainProcessor<MyData>
{
    public async Task<MyData> Handle(MyData data, CancellationToken token = default)
    {
        throw new Exception();          // 1. throw an exception
        return Task.FromResult(data);
    }
    
    public FuncDelegate<MyData> CompensateOnFailure { get; set; } = (context, token) =>
    {
        // 2. If an exception occurs anywhere in the execution chain, the compensation mechanism method will be executed.
        return Task.FromResult(context);
    }; 
}

Specify Execution ⚙️

You can specify the order of execution of processors in the chain.

    var result = await CoRProcessor<NumberContext>
            .New()
            .AddRange([
                new AdditionProcessor(),
                new AdditionProcessor(),
                new AdditionProcessor(),
                new AdditionProcessor(),
            ])
            .Execute(new NumberContext()
            {
                Number1 = 1,
                Number2 = 1,
                Operation = Operation.Addition
            }, default, 2, 3); // The processor specified with index 2 and 3 will execute, while the others will not execute

    Console.WriteLine(result.Data);

Create and Execute the Processor Chain 🏗️

You can create and execute a processor chain using the CoRProcessor<T> class. Here's how to do it:

class Program
{
    public class MyData : IChainContext
    {
        public bool Abort { get; set; } // Abort = true, To stop processing
        public string Data { get; set; }
    }
    
    static async Task Main(string[] args)
    {
        var processors = new List<IChainProcessor<MyData>>
        {
            new SampleProcessor(),
            new AnotherProcessor() // Another processor implementing IChainProcessor<MyData>
        };

        var processor = CoRProcessor<MyData>.New()
            .AddRange(processors)
            .GlobalPreExecute(async (data, token) =>
            {
                Console.WriteLine("Before action");
                await Task.CompletedTask;
            })
            .GlobalExecuted(async (data, token) =>
            {
                Console.WriteLine("After action");
                await Task.CompletedTask;
            })
            .Finally(async (data, token) =>
            {
                Console.WriteLine("Finally action");
                await Task.CompletedTask;
            })
            .OnException(async (data, token) =>
            {
                Console.WriteLine("Exception occurred");
                await Task.FromResult(false); // Returning false will not throw an exception.
            });

        var result = await processor.Execute(new MyData(), CancellationToken.None);
    }

Methods 📚

  • New(): Creates a new instance of the CoRProcessor<T>.
  • AddRange(IEnumerable<IChainProcessor> processors): Adds a range of processors to the chain.
  • Execute(T t, CancellationToken token = default): Executes the processor chain with the provided data and cancellation token.
  • Before(FuncDelegate action): Adds an action to be executed before the main processing.
  • After(FuncDelegate action): Adds an action to be executed after the main processing.
  • Finally(FuncDelegate action): Adds an action to be executed after all processing is complete.
  • OnException(FuncDelegate action): Adds an action to be executed when an exception occurs.

Exception Handling ⚠️

To handle exceptions, you can use the OnException method. This allows you to specify an action that should be taken when an exception occurs during processing.

processor.OnException(async (data, token) =>
{
    Console.WriteLine("Exception occurred");
    await Task.FromResult(false); // Returning false will not throw an exception.
    await Task.FromResult(true);  // Returning true will throw an exception.
});

Dependency Injection (DI) Usage☀️

Microsoft Dependency Injection (DI) 🏢

You can integrate the CoRProcessor with Microsoft's built-in Dependency Injection (DI) system. Here's an example of how to do it: As long as IChainProcessoris implemented, the AddCoR method will automatically register

Configure DI in your Console Application:

class Program
{
    static async Task Main(string[] args)
    {
        var serviceProvider = new ServiceCollection()
            .AddCoR(typeof(Program).Assembly)
            .BuildServiceProvider();

        var additionProcessor = serviceProvider.GetRequiredService<AdditionProcessor>();

        var result = await CoRProcessor<NumberContext>
            .New()
            .AddRange(new[] { additionProcessor })
            .Execute(new NumberContext
            {
                Number1 = 1,
                Number2 = 1,
                Operation = Operation.Addition
            }, default);
        
        Console.WriteLine(result);
    }
}

Autofac Integration 🛠️

You can also use Autofac for Dependency Injection. Here’s how you can integrate Autofac with the CoRProcessor framework: As long as IChainProcessoris implemented, the AddCoR method will automatically register.

Configure Autofac in your Console Application:

class Program
{
    static async Task Main(string[] args)
    {
        var builder = new ContainerBuilder();
        var container = builder.AddCoR(typeof(UnitTests).Assembly).Build();

        var additionProcessor = container.Resolve<AdditionProcessor>();

        var result = await CoRProcessor<NumberContext>
            .New()
            .AddRange([
                additionProcessor
            ])
            .Execute(new NumberContext()
            {
                Number1 = 1,
                Number2 = 1,
                Operation = Operation.Addition
            }, default);
        
        Console.WriteLine(result);
    }
}

Practical examples ☀️

Here is an example of how to use the CoRProcessor framework in a practical scenario:

 var result = await CoRProcessor<InsertOrUpdateOrderProcessorContext>
            .New()
            .AddRange([
                orderPreProcessor,
                orderValidaProcessor,
                orderCustomerProcessor,
                subTotalCalculationProcessor,
                discountAndChargeCalculationProcessor,
                subTotalBeforeDiscountCalculationProcessor,
                taxCalculationBeforeDiscountProcessor,
                taxCalculationAfterDiscountedProcessor,
                tipsCalculationProcessor,
                saveOrderRelationProcessor
            ])
            .Execute(new()
            {
                Merchant = merchant,
                Order = order,
            }, cancellationToken).ConfigureAwait(false);

License 📄

This project is licensed under the MIT License. See the LICENSE file for details.