Interceptors allow you to customize the behaviour of cljs-ajax
. Examples of why you might want to do this include:
- Adding a session token to every request
- Special casing
DELETE
on Google App Engine - Treating empty responses as
null
with JSON responses
There will be example implementations of each of these nearer the bottom of the file.
The request and response formats are implementing using interceptors, so you've got access to everything possible. An interceptor is any object that implements the ajax.core/Interceptor
protocol. The easiest way to achieve this is to call to-interceptor
passing a map with the following keys
:request
- The request interceptor, optional:response
- The response interceptor, also optional:name
- The name of the interceptor, optional but highly recommended for debugging purposes.
A request works like this:
- If you're using the easy API,
GET
orPOST
&c, the request map is converted to a form accepted byajax-request
. - The
ajax-request
map has its method normalized, so:get
becomes"GET"
&c - The response format is determined and fixed
- If no interceptors are provided, they are replaced with
@default-interceptors
.default-interceptors
is an atom vector of interceptors. - Standard interceptors are put around them:
:response-format
at the start and:request-format
at the end. - The request interceptors are run. By the end, the request should have a
:body
entry (unless it's aGET
). - The ajax query is run, it returns an
AjaxResponse
. - The response interceptors are run in reverse order. This means
:response-format
interceptor runs last. - The
:handler
is called with the result of that.
Note that request and response handlers are run using reduce
. This means that you can use reduced
to skip the rest of the interceptors
. Bear in mind that in the case of a response interceptor, this means you need to convert to [ok result]
format before calling reduced
.
You can use an interceptor to change the request format, but not the response format. This is partly caused by technical internal considerations, but it's worth considering that most of the reasons you'd want it are either to slightly customize response behaviour, which can be handled with an interceptor, or to switch formats, which can be handled with format detection.
Equally, you can't use an interceptor to change the list of interceptors. This would be technically feasible if complex, but I can't think of any cases where it wouldn't be a code smell. In general terms, interceptors should be independent of one another. If they're not, it's probably best to just create a control interceptor that calls the others rather than trying to 'trick' the architecture to achieve the same ends.
Quite a few people want to put CSRF or session tokens into every request.
(def token (atom 2334))
(def token-interceptor
(to-interceptor {:name "Token Interceptor"
:request #(assoc-in % [:params :token] @token)}))
(swap! default-interceptors (partial cons token-interceptor))
Note that in general, we will want to prepend elements to the default-interceptor
atom as we do above.
We may wish to see a rule appended, however. The Google App Engine doesn't permit a body in DELETE
requests; its implementation of the HTTP spec is wrong, but there's no point arguing. The following appends this rule:
(defn delete-is-empty [{:keys [method] :as request}]
(if (= method "DELETE")
(reduced (assoc request :body nil))
request))
(def app-engine-delete-interceptor
(to-interceptor {:name "Google App Engine Delete Rule"
:request delete-is-empty}))
;;; Since this rule uses `reduced`, it is important that it is
;;; positioned at the end of the list, hence we use `concat` here
(swap! standard-interceptors concat [app-engine-delete-interceptor])
Finally, many systems return an empty JSON response when returning nil
. Strictly speaking, they should return "null"
.
(defn empty-means-nil [response]
(if (ajax.protocols/-body response)
response
(reduced [(-> response ajax.protocols/-status ajax.core/success?) nil])))
(def treat-nil-as-empty
(to-interceptor {:name "JSON special case nil"
:response empty-means-nil}))
(swap! standard-interceptors concat [treat-nil-as-empty]))