In the last posts, I’ve presented the new ASP.NET Web API processing architecture and described three different hosting capabilities, supported “out of the box”: web hosting, in-memory hosting and self-hosting.
In this post, I will describe the development of a custom host using the Azure Service Bus relaying capabilities. This new host enables the exposure of a Web API on the public cloud, while running on a private machine (e.g. my laptop), that is, a machine without inbound connectivity (e.g. private addresses, firewall, NAT).
This host design is inspired in the self-host architecture, namely the usage of WCF and its integration with the service bus and WCF. Is composed by the following main components, shown in the following diagram.
- The HttpServiceBusConfiguration class derives from HttpConfiguration and adds a couple of properties specific to the this scenario, such as the bus authentication credentials (IssuerName and IssuerSecret).
- The HttpServiceBusServer is initialized with a HttpServiceBusConfiguration and internally performs the following:
- Creates an inner HttpServer instance, using the given configuration.
- Creates a WebServiceHost, using the generic DispatcherService as the service class.
- Adds an endpoint with the WebHttpRelayBinding binding and a TransportClientEndpointBehavior, which configures this endpoint with the credentials required when listening on the bus.
- When a request message is received, the WCF runtime delivers it to the DispatcherService, containing generic asynchronous operations to handle GET requests (BeginGet and EndGet methods) and other HTTP methods (BeginInvoke and EndInvoke).
[ServiceContract] [ServiceBehavior(InstanceContextMode = InstanceContextMode.Single, ConcurrencyMode = ConcurrencyMode.Multiple)] internal class DispatcherService { private readonly HttpServer _server; private readonly HttpServiceBusConfiguration _config; public DispatcherService(HttpServer server, HttpServiceBusConfiguration config) { _server = server; _config = config; } [WebGet(UriTemplate = "*")] [OperationContract(AsyncPattern = true)] public IAsyncResult BeginGet(AsyncCallback callback, object state) { var context = WebOperationContext.Current; return DispatchToHttpServer(context.IncomingRequest, null, context.OutgoingResponse, _config.BufferRequestContent, callback, state); } public Message EndGet(IAsyncResult ar) { var t = ar as Task; var stream = t.Result; return StreamMessageHelper.CreateMessage(MessageVersion.None, "GETRESPONSE", stream ?? new MemoryStream()); } [WebInvoke(UriTemplate = "*", Method = "*")] [OperationContract(AsyncPattern = true)] public IAsyncResult BeginInvoke(Stream s, AsyncCallback callback, object state) { var context = WebOperationContext.Current; return DispatchToHttpServer(context.IncomingRequest, s, context.OutgoingResponse, _config.BufferRequestContent, callback, state); } public Message EndInvoke(IAsyncResult ar) { var t = ar as Task; var stream = t.Result; return StreamMessageHelper.CreateMessage(MessageVersion.None, "GETRESPONSE", stream ?? new MemoryStream()); } ... }
- These generic operations convert the WCF requests, represented by the older IncomingWebRequestContext class, into instances of the new HttpRequestMessage class. Then, they pushe these messages into the HttpServer pipeline. When the server finally returns the responses’ HttpResponseMessage instances, the generic operations convert them back into WCF messages.
The code, still in alpha/”works in my machine” status, is available from https://github.com/pmhsfelix/WebApi.Explorations.ServiceBusRelayHost.
Feedback is appreciated.
