Content-type negotiation and method dispatch in Clojure
Christophe Rhodes’ post on http-content-negotiation and generalized specializers in CLOS (Common Lisp Object System) made an ugliness in a small Clojure web application jump right into my face. I’m using liberator to setup so-called resources (side-note: While this post assumes some familiarity with liberator, the main aspects is actually multi-method handling in Clojure — I hope it’s useful even if you don’t know or care about liberator). Resources are serving as ring handlers (typically used with compojure) and are used to deal with most aspects of request handling in a fairly declarative manner, including content negotiation. Liberator provides decision points and handlers, moving a so-called context around between the various functions that you need to associate with resources — map-like data returned from a decision function will be merged with the existing context. So far, so good. The bad part, however, was that I used a single resource definition for providing multiple media types. More exactly my code / resource definition has an anonymous handler function which uses a simple value check to serve the correct media type (we’re talking about the Accept-Header of the incoming request, cf. RfC 2616, Sec. 14.1, like this (as you can imagine, that’s a somewhat simplified version):
(defresource users
:available-media-types ["text/html" "application/json"]
:method-allowed? (request-method-in :get)
:exists? (fn [context]
{:users (find-users)})
:handle-ok (fn [context]
(let [media-type
(get-in context [:representation :media-type])]
(condp = media-type
"application/json"
(generate-string (get context :users))
"text/html"
(usersview (:users context)))))
:handle-not-found (fn [context]
(let [media-type
(get-in context [:representation :media-type])]
;; TODO: Handle not found for HTML
(condp = media-type
"application/json"
(generate-string {:error "No such user"})))))
From a functional point of view, there is not much wrong with this. It’s very close to the description in the relevant part of the liberator tutorial on content negotiation. But from an aesthetical point of view, the condp
expressions to determine finally how to present the resource data is plainly ugly. To get rid of this ugliness, the inspiration I took from Christophe’s article is to rely on Clojures method dispatch (which is the simple part from Christophe’s post only).
The idea is straight-forward: Instead of using a simple anonymous function which convolutes two different media-types, introduce a multi-method like users-handle-ok
that dispatches on media-type: We can simply define a dispatch method (via defmulti
), moving the code which determines the accepted media-type (e.g. “application/json”). This value is then used by Clojure to determine the right method to use.
(defmulti users-handle-ok
"Handle OK for users resource for different media-types"
(fn [context]
(get-in context [:representation :media-type])))
(defmethod users-handle-ok "application/json" [context]
(generate-string (get context :users)))
(defmethod users-handle-ok "text/html" [context]
(usersview (get context :users)))
;; some code elided here ...
(defresource users
:available-media-types ["text/html" "application/json"]
:method-allowed? (request-method-in :get)
:exists? #(users-exists? %)
:handle-ok #(users-handle-ok %)
:handle-not-found #(users-handle-not-found %))
From a clean code perspective, this has two benefits: we now have mainly code left which does one thing at a time (SRP), which is what we should aim for and which makes unit testing also somewhat easier and more to the point. It also slims down the amount of code in the resource definition considerably. It’s now much more obvious that the resource definition is (from the application developer point of view) not much more than an integration point for different other functions.
Of course, we can use a similar approach for all of the other methods as well. Let’s assume that I have a resource that can generate HTML and JSON, but expects that all incoming POST requests contain JSON only. This will look utterly similar to the approach above, only this time we dispatch on the request-method
. If we are now POST-ing to this resource with a different Content-Type, we’ll receive a “415 Unsupported media type” reply from liberator.
(defmulti known-content-type?
"Determine known content types depending on request-method"
(fn [context]
(get-in context [:request :request-method])))
(defmethod known-content-type? :post [context]
"Allow only application/json for POST requests"
(when-let [content-type (get-in context [:request :content-type])]
(condp = content-type
"application/json" true
false)))
(defmethod known-content-type? :default [_]
true)
(defresource someresource
:available-media-types ["text/html" "application/json"]
:method-allowed? (request-method-in :get :post)
:known-content-type? #(known-content-type? %)
:exists? (fn [context]
(when-let [data (find-daa)]
{:data data}))
:handle-ok #(handle-ok %)
:post! #(handle-post! %)
:post-redirect? (fn [context] {:location (url-in-context "someurl")}))
As you might guess, this method known-content-type?
is probably applicable for most resources. But how would you handle the exception to the exception? This is actually quite easy, as it turns out. In line with most examples of multi-methods I’ve seen so far, we’ve used a simple value to dispatch on. But of course a map can be a value, too. Given the need to override (specialize) the method for some resource, the idea is to define the dispatch method to return a map with the request-method and the resource. We then define the methods with appropriate values. The nice thing about this is that it’s very easy to arrange for default behavior for a request method by just leaving out the resource key — the dispatch function takes precautions not to add a superfluous :resource
key in case none is added to the context by the resource.
(defmulti known-content-type?
"Determine known content types depending on request-method"
(fn [context]
(logging/info (str "Found resource: " (:resourceclass context)))
(logging/info (str "Method: " (get-in context [:request :request-method])))
(let [dispatchval {:request-method (get-in context [:request :request-method])}]
(if-let [resource (:resourceclass context)]
(assoc dispatchval :resourceclass resource)
dispatchval))))
(defmethod known-content-type? {:request-method :post} [context]
"Allow only application/json for POST requests"
(logging/info "Determining known content-type for :post!")
(when-let [content-type (get-in context [:request :content-type])]
(condp = content-type
"application/json" true
false)))
(defmethod known-content-type? :default [_]
(logging/info "Determining known content-type for :default!")
true)
(defresource some-resource
:available-media-types ["text/html" "application/json"]
:method-allowed? (request-method-in :get :post)
:known-content-type? #(known-content-type? %)
:exists? (fn [context]
(when-let [data (find-daa)]
{:data data}))
:handle-ok #(handle-ok %)
:post! #(handle-post! %)
:post-redirect? (fn [context] {:location (url-in-context "someurl")}))
(defresource special-resource
:available-media-types ["text/html" "application/json"]
:method-allowed? (request-method-in :get :post)
:known-content-type? #(known-content-type? (assoc % :resource special-resource))
:exists? (fn [context]
(when-let [data (find-special-data)]
{:data data}))
:handle-ok #(special-handle-ok %)
:post! #(special-post! %)
:post-redirect? (fn [context] {:location (url-in-context "specials")}))
(defmethod known-content-type? {:request-method :post :resource special-resource} [context]
(logging/info "Determining known content-type for :post and special-resources!")
(when-let [content-type (get-in context [:request :content-type])]
(condp = content-type
"application/json" true
"application/x-www-form-urlencoded" true
false)))
With these definitions in place, the default for POST requests using this known-content-type?
method would be to accept only application/json. However, the special-resource
“overrides” this behavior to also accept regular form data. Posting to the various resources will produce output like the following:
2014-04-14 15:04:31,088 [main] INFO utils - Found resource: liberator.core$resource$fn__3268@688dbd21
2014-04-14 15:04:31,088 [main] INFO utils - Method: :get
2014-04-14 15:04:31,089 [main] INFO utils - Known content-type for :default!
2014-04-14 15:27:14,974 [main] INFO utils - Found resource: liberator.core$resource$fn__3268@688dbd21
2014-04-14 15:27:14,974 [main] INFO utils - Method: :post
2014-04-14 15:27:14,975 [main] INFO utils - Known content-type for :post!
2014-04-14 15:04:31,127 [main] INFO utils - Found resource: liberator.core$resource$fn__3268@688dbd21
2014-04-14 15:04:31,127 [main] INFO utils - Method: :post
2014-04-14 15:04:31,127 [main] INFO utils - Known content-type for :post and special-resources!
Please note that known-content-type?
has to be a known symbol (defined or at least declared) prior to be usable in the resource definition, whereas adding the more specialized method requires the special-resource
to be defined — declaring it won’t be enough.
Using maps as dispatch values seems to be a nice and powerful tool to know about. There are, however, still some points where I see room for improvement:
- We would probably like to use the same mechanism for a ton of functions, always highly similar. E.g. the methods for
handle-not-found
andknown-content-type?
look highly similar on the structural level. Also, when you have multiple resources, the dispatch function for one method type (i.e. something likehandle-ok
) are probably always the same, so are the dispatch arguments (i.e. the media types our web application will handle). Maybe a macro would be useful here, but I haven’t thought it through yet. - Handling the Accept header is actually way more complicated. Fortunately, liberator takes care already of choosing the “right” media-type (cf. again the liberator tutorial on content negotiation. However, as also discussed in the same section of said tutorial, there are more negotiable parameters which might come into play, e.g. language or encoding. This quite obviously could lead to some combinatorial explosion. While the approach using a map outlined above is a way to handle it, this approach is essentially mimicking CLOS’ dispatch on multiple arguments via a single argument dispatch.
- I haven’t even started to think about how one would approach the more advanced problem that Christophe is solving by using his
MOP trickerygeneralized-specializers. - The name of the method
users-handle-ok
isn’t really telling. Of course, a name likeusers2json
orserve-users-view
seem better suited to describe what the respective methods are doing, but this obviously would defeat the idea of using multi-methods and the associated benefits. Still, the name should probably not be tied so close to the resource definition. Using the name parameter of methods is one way to remedy this particular issue. - Finally, apparently the slots of a liberator resource expect function objects. Liberator won’t just take the name of a function and do the right thing, it’ll throw an exception. Not a big deal, given that we might need to mangle the implicit argument (the context) anyway, cf. the
:known-content-type?
slot of thespecial-resource
.