{-# LANGUAGE BangPatterns #-}
{-# LANGUAGE MagicHash #-}
{-# LANGUAGE OverloadedStrings #-}
{-# LANGUAGE Trustworthy #-}
module Data.ByteString.Base32
( encodeBase32
, encodeBase32'
, decodeBase32
, encodeBase32Unpadded
, encodeBase32Unpadded'
, decodeBase32Unpadded
, decodeBase32Padded
, isBase32
, isValidBase32
) where
import qualified Data.ByteString as BS
import Data.ByteString.Internal (ByteString(..))
import Data.ByteString.Base32.Internal
import Data.ByteString.Base32.Internal.Head
import Data.ByteString.Base32.Internal.Tables
import Data.Either (isRight)
import Data.Text (Text)
import qualified Data.Text.Encoding as T
import System.IO.Unsafe (unsafeDupablePerformIO)
encodeBase32 :: ByteString -> Text
encodeBase32 :: ByteString -> Text
encodeBase32 = ByteString -> Text
T.decodeUtf8 (ByteString -> Text)
-> (ByteString -> ByteString) -> ByteString -> Text
forall b c a. (b -> c) -> (a -> b) -> a -> c
. ByteString -> ByteString
encodeBase32'
{-# INLINE encodeBase32 #-}
encodeBase32' :: ByteString -> ByteString
encodeBase32' :: ByteString -> ByteString
encodeBase32' = Addr# -> ByteString -> ByteString
encodeBase32_ Addr#
"ABCDEFGHIJKLMNOPQRSTUVWXYZ234567"#
{-# INLINE encodeBase32' #-}
decodeBase32 :: ByteString -> Either Text ByteString
decodeBase32 :: ByteString -> Either Text ByteString
decodeBase32 bs :: ByteString
bs@(PS ForeignPtr Word8
_ Int
_ !Int
l)
| Int
l Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
== Int
0 = ByteString -> Either Text ByteString
forall a b. b -> Either a b
Right ByteString
bs
| Int
r Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
== Int
0 = IO (Either Text ByteString) -> Either Text ByteString
forall a. IO a -> a
unsafeDupablePerformIO (IO (Either Text ByteString) -> Either Text ByteString)
-> IO (Either Text ByteString) -> Either Text ByteString
forall a b. (a -> b) -> a -> b
$ Int
-> ForeignPtr Word8 -> ByteString -> IO (Either Text ByteString)
decodeBase32_ Int
dlen ForeignPtr Word8
stdDecodeTable ByteString
bs
| Int
r Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
== Int
2 = IO (Either Text ByteString) -> Either Text ByteString
forall a. IO a -> a
unsafeDupablePerformIO (IO (Either Text ByteString) -> Either Text ByteString)
-> IO (Either Text ByteString) -> Either Text ByteString
forall a b. (a -> b) -> a -> b
$ Int
-> ForeignPtr Word8 -> ByteString -> IO (Either Text ByteString)
decodeBase32_ Int
dlen ForeignPtr Word8
stdDecodeTable (ByteString -> ByteString -> ByteString
BS.append ByteString
bs ByteString
"======")
| Int
r Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
== Int
4 = Int
-> ByteString
-> IO (Either Text ByteString)
-> Either Text ByteString
validateLastNPads Int
2 ByteString
bs (IO (Either Text ByteString) -> Either Text ByteString)
-> IO (Either Text ByteString) -> Either Text ByteString
forall a b. (a -> b) -> a -> b
$ Int
-> ForeignPtr Word8 -> ByteString -> IO (Either Text ByteString)
decodeBase32_ Int
dlen ForeignPtr Word8
stdDecodeTable (ByteString -> ByteString -> ByteString
BS.append ByteString
bs ByteString
"====")
| Int
r Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
== Int
5 = Int
-> ByteString
-> IO (Either Text ByteString)
-> Either Text ByteString
validateLastNPads Int
3 ByteString
bs (IO (Either Text ByteString) -> Either Text ByteString)
-> IO (Either Text ByteString) -> Either Text ByteString
forall a b. (a -> b) -> a -> b
$ Int
-> ForeignPtr Word8 -> ByteString -> IO (Either Text ByteString)
decodeBase32_ Int
dlen ForeignPtr Word8
stdDecodeTable (ByteString -> ByteString -> ByteString
BS.append ByteString
bs ByteString
"===")
| Int
r Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
== Int
7 = Int
-> ByteString
-> IO (Either Text ByteString)
-> Either Text ByteString
validateLastNPads Int
5 ByteString
bs (IO (Either Text ByteString) -> Either Text ByteString)
-> IO (Either Text ByteString) -> Either Text ByteString
forall a b. (a -> b) -> a -> b
$ Int
-> ForeignPtr Word8 -> ByteString -> IO (Either Text ByteString)
decodeBase32_ Int
dlen ForeignPtr Word8
stdDecodeTable (ByteString -> ByteString -> ByteString
BS.append ByteString
bs ByteString
"=")
| Bool
otherwise = Text -> Either Text ByteString
forall a b. a -> Either a b
Left Text
"Base32-encoded bytestring has invalid size"
where
!r :: Int
r = Int
l Int -> Int -> Int
forall a. Integral a => a -> a -> a
`rem` Int
8
!q :: Int
q = Int
l Int -> Int -> Int
forall a. Integral a => a -> a -> a
`quot` Int
8
!dlen :: Int
dlen = Int
q Int -> Int -> Int
forall a. Num a => a -> a -> a
* Int
5
{-# INLINE decodeBase32 #-}
encodeBase32Unpadded :: ByteString -> Text
encodeBase32Unpadded :: ByteString -> Text
encodeBase32Unpadded = ByteString -> Text
T.decodeUtf8 (ByteString -> Text)
-> (ByteString -> ByteString) -> ByteString -> Text
forall b c a. (b -> c) -> (a -> b) -> a -> c
. ByteString -> ByteString
encodeBase32Unpadded'
{-# INLINE encodeBase32Unpadded #-}
encodeBase32Unpadded' :: ByteString -> ByteString
encodeBase32Unpadded' :: ByteString -> ByteString
encodeBase32Unpadded' = Addr# -> ByteString -> ByteString
encodeBase32NoPad_ Addr#
"ABCDEFGHIJKLMNOPQRSTUVWXYZ234567"#
{-# INLINE encodeBase32Unpadded' #-}
decodeBase32Unpadded :: ByteString -> Either Text ByteString
decodeBase32Unpadded :: ByteString -> Either Text ByteString
decodeBase32Unpadded bs :: ByteString
bs@(PS ForeignPtr Word8
_ Int
_ !Int
l)
| Int
l Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
== Int
0 = ByteString -> Either Text ByteString
forall a b. b -> Either a b
Right ByteString
bs
| Int
r Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
== Int
0 = Int
-> ByteString
-> IO (Either Text ByteString)
-> Either Text ByteString
validateLastNPads Int
1 ByteString
bs (IO (Either Text ByteString) -> Either Text ByteString)
-> IO (Either Text ByteString) -> Either Text ByteString
forall a b. (a -> b) -> a -> b
$ Int
-> ForeignPtr Word8 -> ByteString -> IO (Either Text ByteString)
decodeBase32_ Int
dlen ForeignPtr Word8
stdDecodeTable ByteString
bs
| Int
r Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
== Int
2 = IO (Either Text ByteString) -> Either Text ByteString
forall a. IO a -> a
unsafeDupablePerformIO (IO (Either Text ByteString) -> Either Text ByteString)
-> IO (Either Text ByteString) -> Either Text ByteString
forall a b. (a -> b) -> a -> b
$ Int
-> ForeignPtr Word8 -> ByteString -> IO (Either Text ByteString)
decodeBase32_ Int
dlen ForeignPtr Word8
stdDecodeTable (ByteString -> ByteString -> ByteString
BS.append ByteString
bs ByteString
"======")
| Int
r Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
== Int
4 = Int
-> ByteString
-> IO (Either Text ByteString)
-> Either Text ByteString
validateLastNPads Int
1 ByteString
bs (IO (Either Text ByteString) -> Either Text ByteString)
-> IO (Either Text ByteString) -> Either Text ByteString
forall a b. (a -> b) -> a -> b
$ Int
-> ForeignPtr Word8 -> ByteString -> IO (Either Text ByteString)
decodeBase32_ Int
dlen ForeignPtr Word8
stdDecodeTable (ByteString -> ByteString -> ByteString
BS.append ByteString
bs ByteString
"====")
| Int
r Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
== Int
5 = Int
-> ByteString
-> IO (Either Text ByteString)
-> Either Text ByteString
validateLastNPads Int
1 ByteString
bs (IO (Either Text ByteString) -> Either Text ByteString)
-> IO (Either Text ByteString) -> Either Text ByteString
forall a b. (a -> b) -> a -> b
$ Int
-> ForeignPtr Word8 -> ByteString -> IO (Either Text ByteString)
decodeBase32_ Int
dlen ForeignPtr Word8
stdDecodeTable (ByteString -> ByteString -> ByteString
BS.append ByteString
bs ByteString
"===")
| Int
r Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
== Int
7 = Int
-> ByteString
-> IO (Either Text ByteString)
-> Either Text ByteString
validateLastNPads Int
1 ByteString
bs (IO (Either Text ByteString) -> Either Text ByteString)
-> IO (Either Text ByteString) -> Either Text ByteString
forall a b. (a -> b) -> a -> b
$ Int
-> ForeignPtr Word8 -> ByteString -> IO (Either Text ByteString)
decodeBase32_ Int
dlen ForeignPtr Word8
stdDecodeTable (ByteString -> ByteString -> ByteString
BS.append ByteString
bs ByteString
"=")
| Bool
otherwise = Text -> Either Text ByteString
forall a b. a -> Either a b
Left Text
"Base32-encoded bytestring has invalid size"
where
!q :: Int
q = Int
l Int -> Int -> Int
forall a. Integral a => a -> a -> a
`quot` Int
8
!r :: Int
r = Int
l Int -> Int -> Int
forall a. Integral a => a -> a -> a
`rem` Int
8
!dlen :: Int
dlen = Int
q Int -> Int -> Int
forall a. Num a => a -> a -> a
* Int
5
{-# INLINE decodeBase32Unpadded #-}
decodeBase32Padded :: ByteString -> Either Text ByteString
decodeBase32Padded :: ByteString -> Either Text ByteString
decodeBase32Padded bs :: ByteString
bs@(PS ForeignPtr Word8
_ Int
_ !Int
l)
| Int
l Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
== Int
0 = ByteString -> Either Text ByteString
forall a b. b -> Either a b
Right ByteString
bs
| Int
r Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
== Int
1 = Text -> Either Text ByteString
forall a b. a -> Either a b
Left Text
"Base32-encoded bytestring has invalid size"
| Int
r Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
== Int
3 = Text -> Either Text ByteString
forall a b. a -> Either a b
Left Text
"Base32-encoded bytestring has invalid size"
| Int
r Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
== Int
6 = Text -> Either Text ByteString
forall a b. a -> Either a b
Left Text
"Base32-encoded bytestring has invalid size"
| Int
r Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
/= Int
0 = Text -> Either Text ByteString
forall a b. a -> Either a b
Left Text
"Base32-encoded bytestring requires padding"
| Bool
otherwise = IO (Either Text ByteString) -> Either Text ByteString
forall a. IO a -> a
unsafeDupablePerformIO (IO (Either Text ByteString) -> Either Text ByteString)
-> IO (Either Text ByteString) -> Either Text ByteString
forall a b. (a -> b) -> a -> b
$ Int
-> ForeignPtr Word8 -> ByteString -> IO (Either Text ByteString)
decodeBase32_ Int
dlen ForeignPtr Word8
stdDecodeTable ByteString
bs
where
!q :: Int
q = Int
l Int -> Int -> Int
forall a. Integral a => a -> a -> a
`quot` Int
8
!r :: Int
r = Int
l Int -> Int -> Int
forall a. Integral a => a -> a -> a
`rem` Int
8
!dlen :: Int
dlen = Int
q Int -> Int -> Int
forall a. Num a => a -> a -> a
* Int
5
{-# INLINE decodeBase32Padded #-}
isBase32 :: ByteString -> Bool
isBase32 :: ByteString -> Bool
isBase32 ByteString
bs = ByteString -> Bool
isValidBase32 ByteString
bs Bool -> Bool -> Bool
&& Either Text ByteString -> Bool
forall a b. Either a b -> Bool
isRight (ByteString -> Either Text ByteString
decodeBase32 ByteString
bs)
{-# INLINE isBase32 #-}
isValidBase32 :: ByteString -> Bool
isValidBase32 :: ByteString -> Bool
isValidBase32 = ByteString -> ByteString -> Bool
validateBase32 ByteString
"ABCDEFGHIJKLMNOPQRSTUVWXYZ234567"
{-# INLINE isValidBase32 #-}