Permalink
Fetching contributors…
Cannot retrieve contributors at this time
96 lines (85 sloc) 3.04 KB
(ns ring.util.codec
"Encoding and decoding utilities."
(:use ring.util.data)
(:require [clojure.string :as str])
(:import java.io.File
(java.net URLEncoder URLDecoder)
org.apache.commons.codec.binary.Base64))
(defn- double-escape [^String x]
(.replace x "\\" "\\\\"))
(defn percent-encode
"Percent-encode every character in the given string using either the specified
encoding, or UTF-8 by default."
[unencoded & [encoding]]
(->> (.getBytes unencoded (or encoding "UTF-8"))
(map (partial format "%%%02X"))
(str/join)))
(defn- parse-bytes [encoded-bytes]
(->> (re-seq #"%.." encoded-bytes)
(map #(subs % 1))
(map #(.byteValue (Integer/parseInt % 16)))
(byte-array)))
(defn percent-decode
"Decode every percent-encoded character in the given string using the
specified encoding, or UTF-8 by default."
[encoded & [encoding]]
(str/replace encoded
#"(?:%..)+"
(fn [chars]
(-> (parse-bytes chars)
(String. (or encoding "UTF-8"))
(double-escape)))))
(defn url-encode
"Returns the url-encoded version of the given string, using either a specified
encoding or UTF-8 by default."
[unencoded & [encoding]]
(str/replace
unencoded
#"[^A-Za-z0-9_~.+-]+"
#(double-escape (percent-encode % encoding))))
(defn url-decode
"Returns the url-decoded version of the given string, using either a specified
encoding or UTF-8 by default. If the encoding is invalid, nil is returned."
[encoded & [encoding]]
(percent-decode encoded encoding))
(defn base64-encode
"Encode an array of bytes into a base64 encoded string."
[unencoded]
(String. (Base64/encodeBase64 unencoded)))
(defn base64-decode
"Decode a base64 encoded string into an array of bytes."
[^String encoded]
(Base64/decodeBase64 (.getBytes encoded)))
(defn form-decode
"Parse parameters from a string into a map."
([^String param-string]
(form-decode param-string "UTF-8"))
([^String param-string encoding]
(reduce
(fn [param-map encoded-param]
(if-let [[_ key val] (re-matches #"([^=]+)=(.*)" encoded-param)]
(assoc+ param-map
(url-decode key encoding)
(url-decode (or val "") encoding))
param-map))
{}
(str/split param-string #"&"))))
(defn form-encode
"Encode parameters from a map into a string."
([param-map]
(form-encode param-map "UTF-8"))
([param-map encoding]
(form-encode (keys param-map)
(vals param-map)
encoding))
([params values encoding]
(str/join #"&"
(map (fn [param value]
(if (vector? value)
(form-encode (repeat (count value) param)
value
encoding)
(str (url-encode param)
"="
(url-encode value))))
params values))))