A Clojure library for à la carte (orthogonal) Ring request matching and routing.
(Calf path is a synonym for Desire path. The Calf-Path is a poem by Sam Walter Foss.)
- Ring has no built-in routing mechanism; Calfpath delivers this essential feature.
- Orthogonality - match URI patterns, HTTP methods or anything in a Ring request.
- Calfpath is fast (benchmarks included) - there is no cost to what you do not use.
- API is available as both dispatch macros and extensible, data-driven routes.
Leiningen dependency: [calfpath "0.7.2"] (requires Clojure 1.7 or later)
Require namespace:
(require '[calfpath.core  :refer [->uri ->method ->get ->head ->options ->patch ->put ->post ->delete]])
(require '[calfpath.route :as r])When you need to dispatch on URI pattern with convenient API:
(defn handler
  [request]
  ;; ->uri is a macro that dispatches on URI pattern
  (->uri request
    "/user/:id*" [id]  (->uri request
                         "/profile/:type/" [type] (->method request
                                                    :get {:status 200
                                                          :headers {"Content-Type" "text/plain"}
                                                          :body (format "ID: %s, Type: %s" id type)}
                                                    :put {:status 200
                                                          :headers {"Content-Type" "text/plain"}
                                                          :body "Updated"})
                         "/permissions/"   []     (->method request
                                                    :get {:status 200
                                                          :headers {"Content-Type" "text/plain"}
                                                          :body (str "ID: " id)}
                                                    :put {:status 200
                                                          :headers {"Content-Type" "text/plain"}
                                                          :body (str "Updated ID: " id)}))
    "/company/:cid/dept/:did/" [cid did] (->put request
                                           {:status 200
                                            :headers {"Content-Type" "text/plain"}
                                            :body "Data"})
    "/this/is/a/static/route"  []        (->put request
                                           {:status 200
                                            :headers {"Content-Type" "text/plain"}
                                            :body "output"})))Calfpath supports data-driven routes where every route is a map of certain keys. Routes are easy to extend and re-purpose. An example low-level route looks like the following (where route-handler is the same as a Ring handler function):
{:matcher uri-matcher :nested [{:matcher get-matcher :handler handler1}
                               {:matcher post-matcher :handler handler2}]}Every route has two required keys - :matcher and :handler/:nested as described below:
| Key | Required? | Description | 
|---|---|---|
| :matcher | Yes | (fn [request]) -> request?returns request on success andnilon failure | 
| :nested | Either | Routes vector - nested match is attempted on this if matcher was successful | 
| :handler | Either | (fn [request]) -> responsereturns Ring response map, like a Ring handler | 
Calfpath comes with utilities to create common matchers, e.g. URI, HTTP method, fallback etc.
See the route examples below:
;; a route-handler is arity-1 (or arity-3 for async) fn, like a ring-handler
(defn list-user-jobs
  [{:keys [user-id] :as request}]
  ...)
(defn app-routes
  "Return a vector of route specs."
  []
  [;; first route has a partial URI match,implied by a trailing '*'
   {:uri "/users/:user-id*" :nested [{:uri "/jobs/"        :nested [{:method :get  :handler list-user-jobs}
                                                                    {:method :post :handler assign-job}]}
                                     {:uri "/permissions/" :method :get :handler permissions-hanler}]}
   {:uri "/orders/:order-id/confirm/" :method :post :handler confirm-order}        ; :uri is lifted over :method
   {:uri "/health/"  :handler health-status}
   {:uri "/static/*" :handler (-> (fn [_] {:status 400 :body "No such file"})      ; static files serving example
                                ;; the following require Ring dependency in your project
                                (ring.middleware.resource/wrap-resource "public")  ; render files from classpath
                                (ring.middleware.file/wrap-file "/var/www/public") ; render files from filesystem
                                (ring.middleware.content-type/wrap-content-type)
                                (ring.middleware.not-modified/wrap-not-modified))}])
;; create a Ring handler from given routes
(def ring-handler
  (-> (app-routes)   ; return routes vector
    r/compile-routes ; turn every map into a route by populating matchers in them
    r/make-dispatcher))- The :matcherkey must be present in a route spec for dispatch.- In practice, other keys (e.g. :uri,:methodetc.) add the:matcherkey
 
- In practice, other keys (e.g. 
- Either :handleror:nestedkey must be present in a route spec.
- A successful match may return an updated request, or the same request, or nil
You need JDK 1.7 or higher during development.
Running tests:
$ lein do clean, test
$ lein with-profile c07 testRunning performance benchmarks:
$ lein do clean, perf-test
$ lein with-profile c07,perf test  # on specified Clojure versionCopyright © 2015-2019 Shantanu Kumar ([email protected], [email protected])
Distributed under the Eclipse Public License either version 1.0 or (at your option) any later version.