If the sight of “WCF” in the title makes you want to click your back button, stop. Just wait. I tell you that I have been there, too. This isn’t that WCF. Well, it is, sort of. If you listened to our last several podcasts, you heard Glenn Block talk about his transition from MEF to the WCF team and that he was going to be working on REST. Well, it’s here. Before you continue reading, I recommend you:
- Download the bits
- Read Glenn’s posts (yes, there is a link in that last one on how to wire up the Razor view engine)
- Watch the RESTFest keynote and video
I know, That’s probably asking a lot. I promise, you won’t be disappointed. The WCF team has done a bang up job on this, and it’s continuing to improve. For my part, I’ve been working on making the transition to F# smoother.
Why am I excited about this? I see the new Web APIs as a fantastic, Rack+Sinatra-like platform for .NET. No, it’s not exactly the same, but it fulfills a very similar function. The new Web APIs give you a to-the-metal experience, with control over everything in a statically-typed environment. (Yes, even the headers!) A lot of the WCF abstraction has been stripped away so that HTTP, the application protocol, is available in its fullness.
“Sounds great,” you say, but I already have MVC for that. True, and yes, you can do JSON services with MVC. However, HTTP is primarily a messaging protocol, which WCF fits nicely. MVC is primarily an abstraction for building web pages in HTML. So again, yes there is overlap. Give it a whirl, however, and you’ll probably start seeing, as so many in the Ruby community, that the simpler abstraction offered by the WCF Web APIs is more efficient for REST services and even web application environments. (More on that last part towards the end of the series.)
Simple, yes, but you can already see the same power offered by Rack. Middleware is just a step around the corner. A simple Sinatra-like DSL would make this trivially easy to compose application parts. The WCF Web APIs will offer a composition mechanism, though that is still in the works.
What do I mean? Well, let’s look at a trivial example:
For completeness, here’s the FuncHost wrapper and other type declarations I’m using:
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| module Main | |
| open System | |
| open System.Collections.Generic | |
| open System.Net | |
| open System.Net.Http | |
| open Microsoft.ServiceModel.Http | |
| open FSharp.Http | |
| [<EntryPoint>] | |
| let main(args) = | |
| let app (request:HttpRequestMessage) = async { | |
| // do some stuff with the request | |
| return "200 OK", Seq.empty, "Howdy!"B |> Seq.map (fun b -> b :> obj) } | |
| let processors = [| (fun op -> new PlainTextProcessor(op, MediaTypeProcessorMode.Response) :> System.ServiceModel.Dispatcher.Processor) |] | |
| let host = new FuncHost(app, responseProcessors = processors, baseAddresses = [|"http://localhost:1000/"|]) | |
| host.Open() | |
| printfn "Host open. Hit enter to exit…" | |
| printfn "Use a web browser and go to %sroot or do it right and get fiddler!" baseurl | |
| System.Console.Read() |> ignore | |
| host.Close() | |
| 0 |
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| namespace FSharp.Http | |
| open System | |
| open System.Collections.Generic | |
| open System.Net | |
| open System.Net.Http | |
| open System.ServiceModel | |
| open System.ServiceModel.Web | |
| // NOTE: This is not the actual OWIN definition of an application, just a close approximation. | |
| type Application = Action<HttpRequestMessage, Action<string, seq<KeyValuePair<string,string>>, seq<obj>>, Action<exn>> | |
| module Owin = | |
| let fromAsync (app:HttpRequestMessage -> Async<string * seq<KeyValuePair<string,string>> * seq<obj>>) : Application = | |
| Action<_,_,_>(fun request (onCompleted:Action<string, seq<KeyValuePair<string,string>>, seq<obj>>) (onError:Action<exn>) -> | |
| Async.StartWithContinuations(app request, onCompleted.Invoke, onError.Invoke, onError.Invoke)) | |
| /// <summary>Creates a new instance of <see cref="Processor"/>.</summary> | |
| /// <param name="onExecute">The function to execute in the pipeline.</param> | |
| /// <param name="onGetInArgs">Gets the incoming arguments.</param> | |
| /// <param name="onGetOutArgs">Gets the outgoing arguments.</param> | |
| /// <param name="onError">The action to take in the event of a processor error.</param> | |
| /// <remarks> | |
| /// This subclass of <see cref="System.ServiceModel.Dispatcher.Processor"/> allows | |
| /// the developer to create <see cref="System.ServiceModel.Dispatcher.Processor"/>s | |
| /// using higher-order functions. | |
| /// </remarks> | |
| type Processor(onExecute, ?onGetInArgs, ?onGetOutArgs, ?onError) = | |
| inherit System.ServiceModel.Dispatcher.Processor() | |
| let onGetInArgs' = defaultArg onGetInArgs (fun () -> null) | |
| let onGetOutArgs' = defaultArg onGetOutArgs (fun () -> null) | |
| let onError' = defaultArg onError ignore | |
| override this.OnGetInArguments() = onGetInArgs'() | |
| override this.OnGetOutArguments() = onGetOutArgs'() | |
| override this.OnExecute(input) = onExecute input | |
| override this.OnError(result) = onError' result | |
| /// <summary>Creates a new instance of <see cref="FuncConfiguration"/>.</summary> | |
| /// <param name="requestProcessors">The processors to run when receiving the request.</param> | |
| /// <param name="responseProcessors">The processors to run when sending the response.</param> | |
| type FuncConfiguration(?requestProcessors, ?responseProcessors) = | |
| inherit Microsoft.ServiceModel.Http.HttpHostConfiguration() | |
| // Set the default values on the optional parameters. | |
| let requestProcessors' = defaultArg requestProcessors Seq.empty | |
| let responseProcessors' = defaultArg responseProcessors Seq.empty | |
| // Allows partial application of args to a function using function composition. | |
| let create args f = f args | |
| interface Microsoft.ServiceModel.Description.IProcessorProvider with | |
| member this.RegisterRequestProcessorsForOperation(operation, processors, mode) = | |
| requestProcessors' |> Seq.iter (processors.Add << (create operation)) | |
| member this.RegisterResponseProcessorsForOperation(operation, processors, mode) = | |
| responseProcessors' |> Seq.iter (processors.Add << (create operation)) | |
| /// <summary>Creates a new instance of <see cref="AppResource"/>.</summary> | |
| /// <param name="app">The application to invoke.</param> | |
| /// <remarks>The <see cref="AppResource"/> serves as a catch-all handler for WCF HTTP services.</remarks> | |
| [<ServiceContract>] | |
| [<ServiceBehavior(InstanceContextMode = InstanceContextMode.Single)>] | |
| type AppResource(app:Application) = | |
| let matchStatus (status:string) = | |
| let statusParts = status.Split(' ') | |
| let statusCode = statusParts.[0] | |
| Enum.Parse(typeof<HttpStatusCode>, statusCode) :?> HttpStatusCode | |
| let handle (request:HttpRequestMessage) (response:HttpResponseMessage) = | |
| app.Invoke(request, | |
| Action<_,_,_>(fun status headers body -> | |
| response.StatusCode <- matchStatus status | |
| response.Headers.Clear() | |
| headers |> Seq.iter (fun (KeyValue(k,v)) -> response.Headers.Add(k,v)) | |
| response.Content <- new ByteArrayContent(body |> Seq.map (fun o -> o :?> byte) |> Array.ofSeq)), | |
| Action<_>(fun e -> Console.WriteLine(e))) | |
| /// <summary>Invokes the application with the specified GET <paramref name="request"/>.</summary> | |
| /// <param name="request">The <see cref="HttpRequestMessage"/>.</param> | |
| /// <returns>The <see cref="HttpResponseMessage"/>.</returns> | |
| /// <remarks>Would like to merge this with the Invoke method, below.</remarks> | |
| [<OperationContract>] | |
| [<WebGet(UriTemplate="*")>] | |
| member x.Get(request, response:HttpResponseMessage) = handle request response | |
| /// <summary>Invokes the application with the specified GET <paramref name="request"/>.</summary> | |
| /// <param name="request">The <see cref="HttpRequestMessage"/>.</param> | |
| /// <returns>The <see cref="HttpResponseMessage"/>.</returns> | |
| [<OperationContract>] | |
| [<WebInvoke(UriTemplate="*", Method="POST")>] | |
| member x.Post(request, response:HttpResponseMessage) = handle request response | |
| /// <summary>Invokes the application with the specified GET <paramref name="request"/>.</summary> | |
| /// <param name="request">The <see cref="HttpRequestMessage"/>.</param> | |
| /// <returns>The <see cref="HttpResponseMessage"/>.</returns> | |
| [<OperationContract>] | |
| [<WebInvoke(UriTemplate="*", Method="PUT")>] | |
| member x.Put(request, response:HttpResponseMessage) = handle request response | |
| /// <summary>Invokes the application with the specified GET <paramref name="request"/>.</summary> | |
| /// <param name="request">The <see cref="HttpRequestMessage"/>.</param> | |
| /// <returns>The <see cref="HttpResponseMessage"/>.</returns> | |
| [<OperationContract>] | |
| [<WebInvoke(UriTemplate="*", Method="DELETE")>] | |
| member x.Delete(request, response:HttpResponseMessage) = handle request response | |
| /// <summary>Invokes the application with the specified GET <paramref name="request"/>.</summary> | |
| /// <param name="request">The <see cref="HttpRequestMessage"/>.</param> | |
| /// <returns>The <see cref="HttpResponseMessage"/>.</returns> | |
| [<OperationContract>] | |
| [<WebInvoke(UriTemplate="*", Method="*")>] | |
| member x.Invoke(request, response:HttpResponseMessage) = handle request response | |
| /// <summary>Creates a new instance of <see cref="FuncHost"/>.</summary> | |
| /// <param name="app">The application to invoke.</param> | |
| /// <param name="requestProcessors">The processors to run when receiving the request.</param> | |
| /// <param name="responseProcessors">The processors to run when sending the response.</param> | |
| /// <param name="baseAddresses">The base addresses to host (defaults to an empty array).</param> | |
| type FuncHost(app, ?requestProcessors, ?responseProcessors, ?baseAddresses) = | |
| inherit System.ServiceModel.ServiceHost(AppResource(app), defaultArg baseAddresses [||]) | |
| let requestProcessors = defaultArg requestProcessors Seq.empty | |
| let responseProcessors = defaultArg responseProcessors Seq.empty | |
| let baseUris = defaultArg baseAddresses [||] | |
| let config = new FuncConfiguration(requestProcessors, responseProcessors) | |
| do for baseUri in baseUris do | |
| let endpoint = base.AddServiceEndpoint(typeof<AppResource>, new HttpMessageBinding(), baseUri) | |
| endpoint.Behaviors.Add(new Microsoft.ServiceModel.Description.HttpEndpointBehavior(config)) | |
| /// <summary>Creates a new instance of <see cref="FuncHost"/>.</summary> | |
| /// <param name="app">The application to invoke.</param> | |
| /// <param name="requestProcessors">The processors to run when receiving the request.</param> | |
| /// <param name="responseProcessors">The processors to run when sending the response.</param> | |
| /// <param name="baseAddresses">The base addresses to host (defaults to an empty array).</param> | |
| new (app: HttpRequestMessage -> Async<string * seq<KeyValuePair<string,string>> * seq<obj>>, ?requestProcessors, ?responseProcessors, ?baseAddresses) = | |
| let baseUris = defaultArg baseAddresses [||] |> Array.map (fun baseAddress -> Uri(baseAddress)) | |
| new FuncHost(Owin.fromAsync app, ?requestProcessors = requestProcessors, ?responseProcessors = responseProcessors, baseAddresses = baseUris) |
Go try it out. Let the team know what you think! If you want to show off what you are doing, consider sending a pull request to http://wcfhttpcontrib.codeplex.com/, where we’ll be adding middleware and other common components for your convenience. More on middleware next time.