OSK.Transmissions.Abstractions 0.3.0

There is a newer version of this package available.
See the version list below for details.
dotnet add package OSK.Transmissions.Abstractions --version 0.3.0                
NuGet\Install-Package OSK.Transmissions.Abstractions -Version 0.3.0                
This command is intended to be used within the Package Manager Console in Visual Studio, as it uses the NuGet module's version of Install-Package.
<PackageReference Include="OSK.Transmissions.Abstractions" Version="0.3.0" />                
For projects that support PackageReference, copy this XML node into the project file to reference the package.
paket add OSK.Transmissions.Abstractions --version 0.3.0                
#r "nuget: OSK.Transmissions.Abstractions, 0.3.0"                
#r directive can be used in F# Interactive and Polyglot Notebooks. Copy this into the interactive tool or source code of the script to reference the package.
// Install OSK.Transmissions.Abstractions as a Cake Addin
#addin nuget:?package=OSK.Transmissions.Abstractions&version=0.3.0

// Install OSK.Transmissions.Abstractions as a Cake Tool
#tool nuget:?package=OSK.Transmissions.Abstractions&version=0.3.0                

OSK.Transmissions

These sets of projects aim to help facilitate sending messages, or transmissions, across a variety of message buses. By using an abstraction layer, consuming projects do not need to know the entire process that is required to fully setup and handle communication to and from a bus. Instead, by using a common abstraction, messages can be broadcasted across the entire set of implemented message buses as needed.

OSK.Transmissions.Abstractions

Provides a set of abstraction/top level classes that can be used for dependency reference without adding a dependency on the core implementation. Consumers wanting to send messages to their message buses will need to utilize the IMessageBroadcaster. This interface is the only necessary object for sending messages, assuming one or more message buses have been added to the dependency container. When sending a transmission, users can target a specific message bus for their message to be sent to or can send to all of the integrated buses in the dependency container. By default, the transmissions are sent to all available message buses.

OSK.Transmissions.s.Abstractions

Provides a set of abstraction/top level classes for message management. The core class used by the projects is the IMessage, which provides a set of common properties all message transmissions will have available. A couple of concrete helper abstraction classes are also available to make implementing messages a little easier.

OSK.Transmissions.Application

Applications that need a message bus running in the background, so that processes can run asynchronously, will need a way to manage the receivers. The Application project provides a simple hosted service that will be added to the dependency container and can run in the background. For custom management, users can either manage their receivers directly by requesting the list of IMessageReceiverGroupBuilder which will allow creating the receivers as needed or by using the IMessageReceiverManager to start or stop their receivers as needed

Usage: Message Bus Integrations

To use different message bus integrations, consumers will want to implement or add an implementation that utilizes the library in the dependency container. To do this, consumers will need to ensure the core services are added via one of the AddTransmissions extensions. Transmissions are sent via transmitters to their listening receivers, and how the transmissions are handled can be unique per transmission channel or shared across global configuration middlewares. To configure globally shared setups, users can use the AddTransmissions overload that provides a configuration action for TransmissionsConfigurationOptions, which will be used when building the receivers.

Once core services have been added, users can add a message bus to their dependency container via the AddMessageTransmitter extension. This extension provides a way to add message transmitters, or message buses, along with their unique receivers that will listen for the transmissions that are sent. This is a strongly typed extension to help with understanding what message bus is being added to the dependency container at a glance. When adding a transmitter/receiver configuration, a transmitter id must be provided so that transmission broadcasts will have a way to target specific message buses. These transmitter ids must be unique. IMessageTransmitter implementations will provide the core implementation for how to communicate with the actual message bus, such as RabbitMQ, Kaefka, Local, or other message buses and should be created with any implementation.

Along with a core transmitter integration, integrations will need to implement an associated IMessageReceiver. These will be what listen for transmissions that are sent via a transmitter. These will be started and stopped either by the provided IMessageReceiverManager or some other management mechanism to start and stop each receiver. Receivers should be disposed of when stopped and the IMessageReceiverGroupBuilder is used to create new receivers whenever the manager is started. Implementations of IMessageReceiver should expect several arguments to be provided by the builder in their constructors: a unique string receiverId, a MessageTransmissionDelegate, and an IServiceProvider. A base implementation that can help with integrations is provided with MessageReceiverBase. The passed in delegate should be utilized when a transmission is received from a transmitter and allows middlewares to interact with the transmission until a final handler is used to fully process the message for an application. Middlewares and handlers can be easily added using the MessageReceiverBuilderExtensions. When using AddMessageTransmitter, an IMessageReceiverGroupBuilder object will be provided via a lambda that will give access to the consumer to set shared transmission channel middleware and grant access to an IMessageReceiverBuilder that is used to set the custom handlers and middlewares specific to a receiver.

Construction of the transmitter and receiver objects occurs as needed via internal MessageTransmitterDescriptor and MessageReceiverDescriptor objects that are created for integration via the library extensions when adding new message buses.

While specific implementation configuration for a specific message bus transmitter can vary, below is a generic example of how to create an implementation for a transmitter/receiver integration:

  1. Add a custom message bus transmitter
public class CustomTransmissionsTransmitter: IMessageTransmitter 
{
   public Task TransmitAsync(message, transmissionOptions, cancellationToken) 
   {
     // Custom message bus logic to transmit the message to the server/local machine/etc.
   }
}
  1. Add a message bus receiver for the transmitter
public class CustomTransmissionsReceiver(receiverId, delegate, serviceProvider, customArgument1, customerArgument2, ...)
    : MessageReceiver(receiverId, delegate, serviceProvider)
{
    public void Dispose() 
    {
       // Dispose of specific receiver resources    
    }

    public void Start() 
    {
      // Specific receiver logic to listen for message bus transmitter transmissions
    }
}
  1. Add the transmitter/receiver pair to the dependency container
public static ServiceCollectionExtensions 
{
    public static IServiceCollection AddCustomTransmissions(services) 
    {
        // Ensure a message bus has been added
        services.AddTransmissions(busOptions => {
            busOptions.AddGlobalReceiverConfiguration(receiverBuilder => {  
                // Add globally shared middleware used by all receivers across all message bus transmitters
            })
        })

        services.AddTransmitter<CustomTransmissionsTransmitter, CustomTransmissionsReceiver>("NamedCustomTransmissionsId", groupBuilder => {
           groupBuilder.AddConfigurator(receiverBuilder => {
            // Add shared middleware across all receivers for this transmitter
           })

           groupBuilder.AddMessageReceiver(receiverId, customParameters, receiverBuilder => {
            // Add custom middleware specific to this receiver
           })

           // For instances where an integration may want to implement more than one receiver type, receivers that
           // inherit from the receiver type in the AddTransmitter call can be added as well
           groupBuilder.AddMessageReceiver<CustomReceiverSubClass>(...);
        });
    }
}

Usage: Consumer Broadcasts

After the core library services have been added via any of the AddTransmissions extensions and at least one integration for a message bus has been added, consumers can send messages to their message buses via the IMessageBroadcaster interface. This provides a generic method that allows sending any message type, and a set of broadcast options that can allow setting delayed messages and targetting specific message bus transmitters for transmissions by using their named transmitterId that was provided when adding the transmitter to the dependency container. Broadcasts will attempt to be sent to the target message buses and a BroadcastResult will be returned to provide information to the success of the messages being sent to all the message bus transmitters. For consumers, one of three scenarios can occur and information relating to the result can be determined checking the status code of the output of the broadcaster function call. For quick reference, the 3 scenarios and related responses are as follows:

  • Consumer sends a broadcast that succeeds across all targeted message buses. The response should be HttpStatusCode.OK
  • Consumer sends a broadcast that succeeds across some of the targeted message buses, but fails on others. The response should be HttpStatusCode.MultiStatus
  • Consumer sends a broadcast that fails across all target message buses. The response should be HttpStatusCode.InternalServerError

For all of the above scenarios, the broadcast result will provide exception information for any of the failed transmitters along with their transmitter ids to allow for any retry handling that might be necessary by the consumer.

An example consumer usage might be:

 public class Service(IMessageBroadcaster broadcaster, ...) {
    ...

    public Task DoSomethingAsync() {
       ...

       var broadCastResult = await braodcaster.BroadcastMessageAsync(new CustomMessage(), configuration => {
        configuration.TargetTransmitterIds = [ "CustomBusA", "CustomBusZ" ]
       });


       if (broadCastResult.Code.StatusCode == HttpStatusCode.OK) 
       {
         return;
       }

       var failedTransmissions = broadCastResult.TransmissionResults.Where(transmission => !transmission.Successful);
       
       // Failed transmission logic
       ...
    }
 } 

Extensions for the message broadcaster are available that should hopefully make it easier to use in some cases. These can be found in MessageBroadcasterExtensions

Product 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.  net9.0 was computed.  net9.0-android was computed.  net9.0-browser was computed.  net9.0-ios was computed.  net9.0-maccatalyst was computed.  net9.0-macos was computed.  net9.0-tvos was computed.  net9.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. 
Compatible target framework(s)
Included target framework(s) (in package)
Learn more about Target Frameworks and .NET Standard.

NuGet packages (1)

Showing the top 1 NuGet packages that depend on OSK.Transmissions.Abstractions:

Package Downloads
OSK.Transmissions

A generic implementation layer for communicating across a variety of different message buses seamlessly and easily. Meant to allow easy integration between different message buses without causing drastic changes to the consumers.

GitHub repositories

This package is not used by any popular GitHub repositories.

Version Downloads Last updated
1.0.0 110 1/8/2025
0.3.0 167 12/31/2024