module Network.AWS.Data.Internal.URI
    ( collapseURI
    , encodeURI
    ) where
import           Data.ByteString.Char8       (ByteString)
import qualified Data.ByteString.Char8       as BS
import           Data.Char
import qualified Data.Foldable               as Fold
import           Data.Monoid
import           Network.AWS.Data.Internal.ByteString
collapseURI :: ByteString -> ByteString
collapseURI bs
    | BS.null bs   = slash
    | BS.null path = slash
    | otherwise    = tl (hd path)
  where
    path = BS.intercalate slash
        . reverse
        . Fold.foldl' go []
        . filter (/= mempty)
        $ BS.split sep bs
    hd x | BS.head x  == sep = x
         | otherwise         = sep `BS.cons` x
    tl x | BS.last x  == sep = x
         | BS.last bs == sep = x `BS.snoc` sep
         | otherwise         = x
    go acc c | c == dot  = acc
    go acc c | c == dots = remv acc c
    go acc c             = c : acc
    remv []       _ = []
    remv (x : xs) c
        | x == dot  = c : xs
        | x == dots = c : x : xs
        | otherwise = xs
    dot  = "."
    dots = ".."
    slash = BS.singleton sep
    sep  = '/'
encodeURI :: Bool -> ByteString -> ByteString
encodeURI p = toBS . BS.foldr (mappend . enc) mempty
  where
    enc ' '          = "%20"
    enc c@'/'
        | p          = "%2F"
        | otherwise  = build c
    enc c
        | reserved c = build c
    enc c            = char2hex c
    reserved c =
           isAsciiUpper c
        || isAsciiLower c
        || isDigit c
        || c `elem` "-_.~"
    char2hex c =
        let (a, b) = fromEnum c `divMod` 16
         in build ['%', hex a, hex b]
    hex i | i < 10    = toEnum (48 + i)
          | otherwise = toEnum (65 + i  10)