RabbitRPC.States.EntityFrameworkCore
1.0.0-preview-20220329.1
dotnet add package RabbitRPC.States.EntityFrameworkCore --version 1.0.0-preview-20220329.1
NuGet\Install-Package RabbitRPC.States.EntityFrameworkCore -Version 1.0.0-preview-20220329.1
<PackageReference Include="RabbitRPC.States.EntityFrameworkCore" Version="1.0.0-preview-20220329.1" />
paket add RabbitRPC.States.EntityFrameworkCore --version 1.0.0-preview-20220329.1
#r "nuget: RabbitRPC.States.EntityFrameworkCore, 1.0.0-preview-20220329.1"
// Install RabbitRPC.States.EntityFrameworkCore as a Cake Addin #addin nuget:?package=RabbitRPC.States.EntityFrameworkCore&version=1.0.0-preview-20220329.1&prerelease // Install RabbitRPC.States.EntityFrameworkCore as a Cake Tool #tool nuget:?package=RabbitRPC.States.EntityFrameworkCore&version=1.0.0-preview-20220329.1&prerelease
RabbitRPC
This repository is a work-in-progress.
RabbitRPC is a lightweight .NET communication library for microservices and distributed applications built on top of RabbitMQ.
RabbitRPC aims to simplify the development and deployment of small and mid-sized distributed applications by providing zero configuration communication primitives including remote procedure call, distributed events, shared states and distributed work queues.
RabbitRPC doesn't rely on any microservice infrastructure or development tool. But you can have a smoother development/deployment experience by using Project Tye.
Features
Strongly-typed RPC services and clients
RabbitRPC allows you to setup RPC services inside your ASP.NET Core or standalone console applications with just a few lines of code, zero configuration.
public interface IChatService : IRabbitService
{
Task<string> HelloAsync(string name, CancellationToken cancellationToken = default);
}
class ChatService : IChatService
{
public Task<string> HelloAsync(string name, CancellationToken cancellationToken = default)
{
return Task.FromResult($"Hello {name}!");
}
}
Then you can access the service with a strongly-typed client:
class ClientApp
{
private readonly IChatService _chatService;
public ClientApp(IChatService service)
{
_chatService = service;
}
publice async Task SayHelloAsync(CancellationToken cancellationToken)
{
var greeting = await _chatService.HelloAsync("World", cancellationToken);
Console.WriteLine(greeting);
}
}
No need to worry about service discovery, transport, serialization and contracts. RabbitRPC will handle that for you.
Automatically load-balanced
All RabbitRPC services are by default load-balanced, no configuration is needed. You can run as many replicas as you want, on one computer or across the network. Requests to the service will be evenly distributed to all replicas in a round-robin manner.
Request filtering
RabbitRPC provides a ASP.NET Core MVC like request filtering pipeline, custom cross-cutting concerns can be added to your service actions by defining your own filters.
Currently the following hooks are supported:
OnInitializeServiceInstance
: Prepare service instance to process the request.OnBindParameters
: Read parameters from raw request and bind them to action method parameters.OnActionExecuting
: Called before the action exectuion.OnActionExecuted
: Called after the action execution.
Client proxies also supports request filtering. You can intercept, inspect and modify request and response messages before they are sent or processed.
Client filters provides the following hook timings:
OnPrepareRequest
: Convert method parameters to request message and add custom out-of-band data by using RabbitMQ message properties.OnRequestStarting
: Called before the request is sent to the server.OnResponseReceived
: Called after the response is received from the server.OnRequestCompleted
: Called when the result or error information are extracted from the response.
Shared states
Sometimes you may want to have shared states across multiple replicas. You can achieve this by using a third-party state storage like Redis or a relational database.
RabbitRPC also provides a simple state storage interface which allows you to access different state storage providers. The interface supports transactions and optimistic concurrency control.
To access shared states inside your RPC serivce, simply use StateContext
property:
await StateContext.GetAsync<long>("counter", cancellationToken);
You can also leverage the RetryOnConcurrencyErrorAttribute
action filter to perform concurrency control. The filter will detect any concurrency error occurred during the action execution, and will re-execute the action to retry automatically.
RabbitRPC uses a in-memory provider by default. With the in-memory provider, states cannot be shared across replicas. To use a different provider, you need to register the state context on startup with:
services.AddEntityFrameworkCoreStateContext(options => options.UseSqlite("Data Source=states.db"));
Distributed events
RabbitRPC provides a simple yet powerful strongly-typed event bus, which allows you to publish and subcribe distributed events with low coding overhead.
To publish an event, simply call IRabbitEventBus.Publish
and the event will be sent to all subscribers.
EventBus.Publish(new MyEvent());
To subscribe a event, call IRabbitEventBus.Observe<T>
to retrieve an IObservable<T>
instance for the specific type of event. And you can subscribe the event by calling IObservable<T>.Subscribe
to receive strongly-typed event objects.
EventBus.Observe<MyEvent>().Subscribe(...);
Since the events are streamed through IObservable<T>
, you can also use Reactive Extensions to manipulate and consume the event stream.
Work queues
RabbitRPC implements a distributed durable work queue to support background batch processing. You can run the work item handler along with the RPC service, or in a seperate application.
To utilize background batch processing, you need to define a handler:
class PrintJobHandler : IWorkItemHandler<PrintJob>
{
public async Task ProcessAsync(ReadOnlyMemory<WorkItem<PrintJob>> items, CancellationToken cancellationToken)
{
for(var i=0; i<items.Length; i++)
{
Console.WriteLine(items.Span[i].Value.Text);
items.Span[i].IsDone = true;
}
}
}
Then register this handler with work queue:
services.AddWorkQueue(x=>x.AddHandler<PrintJob, PrintJobHandler>(options=>
{
options.ConcurrencyMode = BatchConcurrencyMode.Shared;
options.BatchTimeout = 1000;
options.BatchSize = 8;
}));
Once the handler is up and running, you can dispatch works by a single line of code:
_workQueue.Post(new PrintJob($"Hello world!"));
Similar to RPC services, work item handlers also support load balancing by default. You can have multiple work item handler which handles same work item type running in different processes or computers.
Getting Started
RabbitRPC is currently under development, you can install preview packages to try out:
dotnet add package RabbitRPC.Core --prerelease
You'll also need to install a serialization provider package using either
dotnet add package RabbitRPC.Serialization.NewtonsoftJson --prerelease
or
dotnet add package RabbitRPC.Serialization.MessagePack --prerelease
To share states between replicas, you can install a EntityFrameworkCore state storage provider which supports using SQLite, Microsoft SQL Server and PostgreSQL to store shared states:
dotnet add package RabbitRPC.States.EntityFrameworkCore --prerelease
Please refer documents in samples diretory for detailed usage information.
Product | Versions Compatible and additional computed target framework versions. |
---|---|
.NET | net5.0 was computed. net5.0-windows was computed. net6.0 was computed. net6.0-android was computed. net6.0-ios was computed. net6.0-maccatalyst was computed. net6.0-macos was computed. net6.0-tvos was computed. net6.0-windows was computed. net7.0 was computed. net7.0-android was computed. net7.0-ios was computed. net7.0-maccatalyst was computed. net7.0-macos was computed. net7.0-tvos was computed. net7.0-windows was computed. net8.0 was computed. 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. |
.NET Core | netcoreapp3.0 was computed. netcoreapp3.1 was computed. |
.NET Standard | netstandard2.1 is compatible. |
MonoAndroid | monoandroid was computed. |
MonoMac | monomac was computed. |
MonoTouch | monotouch was computed. |
Tizen | tizen60 was computed. |
Xamarin.iOS | xamarinios was computed. |
Xamarin.Mac | xamarinmac was computed. |
Xamarin.TVOS | xamarintvos was computed. |
Xamarin.WatchOS | xamarinwatchos was computed. |
-
.NETStandard 2.1
- Microsoft.EntityFrameworkCore (>= 5.0.13)
- Microsoft.EntityFrameworkCore.Relational (>= 5.0.13)
- RabbitRPC.Core (>= 1.0.0-preview-20220329.1)
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 |
---|---|---|
1.0.0-preview-20220329.1 | 143 | 3/29/2022 |
1.0.0-preview-20220328.2 | 133 | 3/28/2022 |
1.0.0-preview-20220328.1 | 128 | 3/28/2022 |
1.0.0-preview-20220114.1 | 141 | 1/14/2022 |