Существует функция url-encode
в пространстве имен Ring * ring.util.codec
:
(ring.util.codec/url-encode "clojure url")
; => "clojure+url"
Я не уверен, есть ли предварительно упакованная функция для преобразования карты в строку запроса, но, возможно, это может помочьзадание:
(use '[ring.util.codec :only [url-encode]])
(defn make-query-string [m]
(->> (for [[k v] m]
(str (url-encode k) "=" (url-encode v)))
(interpose "&")
(apply str)))
Пример:
user> (make-query-string {"q" "clojure url" "foo" "bar"})
"q=clojure+url&foo=bar"
Осталось только объединить результат в конец URL:
(defn build-url [url-base query-map]
(str url-base "?" (make-query-string query-map)))
Кажется, что работает:
user> (build-url "http://stackoverflow.com/search" {"q" "clojure url"})
"http://stackoverflow.com/search?q=clojure+url"
Обновление:
Возможно, модифицированная версия могла бы сделать более дружественным к Clojure.Также обрабатывает кодирование через необязательный аргумент в стиле Ring с utf-8 по умолчанию.
(defn make-query-string [m & [encoding]]
(let [s #(if (instance? clojure.lang.Named %) (name %) %)
enc (or encoding "UTF-8")]
(->> (for [[k v] m]
(str (url-encode (s k) enc)
"="
(url-encode (str v) enc)))
(interpose "&")
(apply str))))
(defn build-url [url-base query-map & [encoding]]
(str url-base "?" (make-query-string query-map encoding)))
Так что теперь мы можем сделать
user> (build-url "http://example.com/q" {:foo 1})
"http://example.com/q?foo=1"