-- | functions for percent encoding/decoding ByteStrings module Percent.Encoder (encode, decode) where import qualified Data.Array as A import qualified Data.Array.Base as AB import Data.Word (Word8) import qualified Data.ByteString as B import qualified Data.ByteString.Char8 as C -- | Percent encode a ByteString encode :: B.ByteString -> B.ByteString encode = B.concatMap encodeByte encodeByte :: Word8 -> B.ByteString encodeByte w = AB.unsafeAt encoderArray $ fromIntegral w -- | Percent decode a ByteString decode :: B.ByteString -> B.ByteString decode bs | B.null bs = B.empty | w1 == 37 = B.cons (w2*16+w3) $ decode t3 -- 37 == % | otherwise = B.cons w1 $ decode t1 where t1 = B.tail bs t2 = B.tail t1 t3 = B.tail t2 w1 = B.head bs w2 = hexToNum $ B.head t1 w3 = hexToNum $ B.head t2 -- adapted from Data.Char.digitToInt hexToNum :: Word8 -> Word8 hexToNum c | dec <= 9 = dec | hexl <= 5 = hexl + 10 | otherwise = hexu + 10 where dec = c - 48 -- 0 hexu = c - 65 -- A hexl = c - 97 -- a encodeList = map C.pack ["%00","%01","%02","%03","%04","%05","%06","%07","%08","%09","%0A","%0B","%0C","%0D","%0E","%0F","%10","%11","%12","%13","%14","%15","%16","%17","%18","%19","%1A","%1B","%1C","%1D","%1E","%1F","%20","%21","%22","%23","%24","%25","%26","%27","%28","%29","%2A","%2B","%2C","-",".","%2F","0","1","2","3","4","5","6","7","8","9","%3A","%3B","%3C","%3D","%3E","%3F","%40","A","B","C","D","E","F","G","H","I","J","K","L","M","N","O","P","Q","R","S","T","U","V","W","X","Y","Z","%5B","%5C","%5D","%5E","_","%60","a","b","c","d","e","f","g","h","i","j","k","l","m","n","o","p","q","r","s","t","u","v","w","x","y","z","%7B","%7C","%7D","~","%7F","%80","%81","%82","%83","%84","%85","%86","%87","%88","%89","%8A","%8B","%8C","%8D","%8E","%8F","%90","%91","%92","%93","%94","%95","%96","%97","%98","%99","%9A","%9B","%9C","%9D","%9E","%9F","%A0","%A1","%A2","%A3","%A4","%A5","%A6","%A7","%A8","%A9","%AA","%AB","%AC","%AD","%AE","%AF","%B0","%B1","%B2","%B3","%B4","%B5","%B6","%B7","%B8","%B9","%BA","%BB","%BC","%BD","%BE","%BF","%C0","%C1","%C2","%C3","%C4","%C5","%C6","%C7","%C8","%C9","%CA","%CB","%CC","%CD","%CE","%CF","%D0","%D1","%D2","%D3","%D4","%D5","%D6","%D7","%D8","%D9","%DA","%DB","%DC","%DD","%DE","%DF","%E0","%E1","%E2","%E3","%E4","%E5","%E6","%E7","%E8","%E9","%EA","%EB","%EC","%ED","%EE","%EF","%F0","%F1","%F2","%F3","%F4","%F5","%F6","%F7","%F8","%F9","%FA","%FB","%FC","%FD","%FE","%FF"] encoderArray = A.listArray (0,255) encodeList