Compojureは「ミドルウェア」を使用して HTTP リクエストを処理します。その実装を確認してください。Compojure の「ハンドラ」は、リクエストを受け取ってレスポンスを返す関数です。(リクエストとレスポンスはどちらも Clojure のハッシュ マップです。) 「ミドルウェア」は、ハンドラー関数を受け取り、別のハンドラー関数を返す関数です。ミドルウェアは、リクエスト、レスポンス、またはその両方を変更できます。渡されたハンドラーを (必要に応じて繰り返し) 呼び出したり、短絡してハンドラーを無視したりできます。この方法で、ハンドラーを他のハンドラーに任意の組み合わせでラップできます。
関数がファーストクラスのオブジェクトであるため、これは非常に軽量で、実装と使用が簡単です。ただし、Java インターフェイスから取得する場合とは異なり、コンパイル時に何も強制しません。それはすべて、慣習とダックタイピングに従うことの問題です。プロトコルは最終的にはこのタスクに適しているかもしれませんが、しばらくは利用できないでしょう (おそらく Clojure 2.0 で?)。
これがあなたが望むものかどうかはわかりませんが、ここに非常に初歩的なバージョンがあります:
;; Handler
(defn default [msg]
{:from "Server"
:to (:from msg)
:response "Hi there."})
;; Middleware
(defn logger [handler]
(fn [msg]
(println "LOGGING MESSAGE:" (pr-str msg))
(handler msg)))
(defn datestamper [handler]
(fn [msg]
(assoc (handler msg)
:datestamp (.getTime (java.util.Calendar/getInstance)))))
(defn short-circuit [handler]
(fn [msg]
{:from "Ninja"
:to (:from msg)
:response "I intercepted your message."}))
;; This would do something with a response (send it to a remote server etc.)
(defn do-something [response]
(println ">>>> Response:" (pr-str response)))
;; Given a message and maybe a handler, handle the message
(defn process-message
([msg] (process-message msg identity))
([msg handler]
(do-something ((-> default handler) msg))))
それで:
user> (def msg {:from "Chester" :to "Server" :message "Hello?"})
#'user/msg
user> (process-message msg)
>>>> Response: {:from "Server", :to "Chester", :response "Hi there."}
nil
user> (process-message msg logger)
LOGGING MESSAGE: {:from "Chester", :to "Server", :message "Hello?"}
>>>> Response: {:from "Server", :to "Chester", :response "Hi there."}
nil
user> (process-message msg (comp logger datestamper))
LOGGING MESSAGE: {:from "Chester", :to "Server", :message "Hello?"}
>>>> Response: {:datestamp #<Date Fri Nov 27 17:50:29 PST 2009>, :from "Server", :to "Chester", :response "Hi there."}
nil
user> (process-message msg (comp short-circuit logger datestamper))
>>>> Response: {:from "Ninja", :to "Chester", :response "I intercepted your message."}
nil