{-# LANGUAGE CPP #-}
{-# LANGUAGE LambdaCase #-}
{-# LANGUAGE OverloadedStrings #-}
module GitHub.REST.Auth (
Token (..),
fromToken,
getJWTToken,
loadSigner,
) where
import Data.ByteString (ByteString)
import qualified Data.ByteString as ByteString
#if !MIN_VERSION_base(4,11,0)
import Data.Monoid ((<>))
#endif
import Data.Text (Text)
import qualified Data.Text as Text
import qualified Data.Text.Encoding as Text
import Data.Time (addUTCTime, getCurrentTime)
import Data.Time.Clock.POSIX (utcTimeToPOSIXSeconds)
import qualified Web.JWT as JWT
#if MIN_VERSION_jwt(0,11,0)
type EncodeSigner = JWT.EncodeSigner
#else
type EncodeSigner = JWT.Signer
#endif
data Token
=
AccessToken ByteString
|
BearerToken ByteString
deriving (Int -> Token -> ShowS
[Token] -> ShowS
Token -> String
(Int -> Token -> ShowS)
-> (Token -> String) -> ([Token] -> ShowS) -> Show Token
forall a.
(Int -> a -> ShowS) -> (a -> String) -> ([a] -> ShowS) -> Show a
showList :: [Token] -> ShowS
$cshowList :: [Token] -> ShowS
show :: Token -> String
$cshow :: Token -> String
showsPrec :: Int -> Token -> ShowS
$cshowsPrec :: Int -> Token -> ShowS
Show)
fromToken :: Token -> ByteString
fromToken :: Token -> ByteString
fromToken = \case
AccessToken ByteString
t -> ByteString
"token " ByteString -> ByteString -> ByteString
forall a. Semigroup a => a -> a -> a
<> ByteString
t
BearerToken ByteString
t -> ByteString
"bearer " ByteString -> ByteString -> ByteString
forall a. Semigroup a => a -> a -> a
<> ByteString
t
type AppId = Int
getJWTToken :: EncodeSigner -> AppId -> IO Token
getJWTToken :: EncodeSigner -> Int -> IO Token
getJWTToken EncodeSigner
signer Int
appId = UTCTime -> Token
mkToken (UTCTime -> Token) -> IO UTCTime -> IO Token
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> IO UTCTime
getNow
where
mkToken :: UTCTime -> Token
mkToken UTCTime
now =
let claims :: JWTClaimsSet
claims =
JWTClaimsSet
forall a. Monoid a => a
mempty
{ iat :: Maybe IntDate
JWT.iat = NominalDiffTime -> Maybe IntDate
JWT.numericDate (NominalDiffTime -> Maybe IntDate)
-> NominalDiffTime -> Maybe IntDate
forall a b. (a -> b) -> a -> b
$ UTCTime -> NominalDiffTime
utcTimeToPOSIXSeconds UTCTime
now
, exp :: Maybe IntDate
JWT.exp = NominalDiffTime -> Maybe IntDate
JWT.numericDate (NominalDiffTime -> Maybe IntDate)
-> NominalDiffTime -> Maybe IntDate
forall a b. (a -> b) -> a -> b
$ UTCTime -> NominalDiffTime
utcTimeToPOSIXSeconds UTCTime
now NominalDiffTime -> NominalDiffTime -> NominalDiffTime
forall a. Num a => a -> a -> a
+ (NominalDiffTime
10 NominalDiffTime -> NominalDiffTime -> NominalDiffTime
forall a. Num a => a -> a -> a
* NominalDiffTime
60)
, iss :: Maybe StringOrURI
JWT.iss = Text -> Maybe StringOrURI
JWT.stringOrURI (Text -> Maybe StringOrURI) -> Text -> Maybe StringOrURI
forall a b. (a -> b) -> a -> b
$ String -> Text
Text.pack (String -> Text) -> String -> Text
forall a b. (a -> b) -> a -> b
$ Int -> String
forall a. Show a => a -> String
show Int
appId
}
in ByteString -> Token
BearerToken (ByteString -> Token) -> (Text -> ByteString) -> Text -> Token
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Text -> ByteString
Text.encodeUtf8 (Text -> Token) -> Text -> Token
forall a b. (a -> b) -> a -> b
$ EncodeSigner -> JWTClaimsSet -> Text
signToken EncodeSigner
signer JWTClaimsSet
claims
getNow :: IO UTCTime
getNow = NominalDiffTime -> UTCTime -> UTCTime
addUTCTime (-NominalDiffTime
1) (UTCTime -> UTCTime) -> IO UTCTime -> IO UTCTime
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> IO UTCTime
getCurrentTime
signToken :: EncodeSigner -> JWT.JWTClaimsSet -> Text
#if MIN_VERSION_jwt(0,10,0)
signToken :: EncodeSigner -> JWTClaimsSet -> Text
signToken = (EncodeSigner -> JOSEHeader -> JWTClaimsSet -> Text)
-> JOSEHeader -> EncodeSigner -> JWTClaimsSet -> Text
forall a b c. (a -> b -> c) -> b -> a -> c
flip EncodeSigner -> JOSEHeader -> JWTClaimsSet -> Text
JWT.encodeSigned JOSEHeader
forall a. Monoid a => a
mempty
#else
signToken = JWT.encodeSigned
#endif
loadSigner :: FilePath -> IO EncodeSigner
loadSigner :: String -> IO EncodeSigner
loadSigner String
file = IO EncodeSigner
-> (EncodeSigner -> IO EncodeSigner)
-> Maybe EncodeSigner
-> IO EncodeSigner
forall b a. b -> (a -> b) -> Maybe a -> b
maybe IO EncodeSigner
forall a. IO a
badSigner EncodeSigner -> IO EncodeSigner
forall (m :: * -> *) a. Monad m => a -> m a
return (Maybe EncodeSigner -> IO EncodeSigner)
-> (ByteString -> Maybe EncodeSigner)
-> ByteString
-> IO EncodeSigner
forall b c a. (b -> c) -> (a -> b) -> a -> c
. ByteString -> Maybe EncodeSigner
readSigner (ByteString -> IO EncodeSigner) -> IO ByteString -> IO EncodeSigner
forall (m :: * -> *) a b. Monad m => (a -> m b) -> m a -> m b
=<< String -> IO ByteString
ByteString.readFile String
file
where
badSigner :: IO a
badSigner = String -> IO a
forall (m :: * -> *) a. MonadFail m => String -> m a
fail (String -> IO a) -> String -> IO a
forall a b. (a -> b) -> a -> b
$ String
"Not a valid RSA private key file: " String -> ShowS
forall a. [a] -> [a] -> [a]
++ String
file
readSigner :: ByteString -> Maybe EncodeSigner
readSigner = (PrivateKey -> EncodeSigner)
-> Maybe PrivateKey -> Maybe EncodeSigner
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap PrivateKey -> EncodeSigner
toEncodeRSAPrivateKey (Maybe PrivateKey -> Maybe EncodeSigner)
-> (ByteString -> Maybe PrivateKey)
-> ByteString
-> Maybe EncodeSigner
forall b c a. (b -> c) -> (a -> b) -> a -> c
. ByteString -> Maybe PrivateKey
JWT.readRsaSecret
#if MIN_VERSION_jwt(0,11,0)
toEncodeRSAPrivateKey :: PrivateKey -> EncodeSigner
toEncodeRSAPrivateKey = PrivateKey -> EncodeSigner
JWT.EncodeRSAPrivateKey
#else
toEncodeRSAPrivateKey = JWT.RSAPrivateKey
#endif