{-# LANGUAGE CPP #-} {-# LANGUAGE OverloadedStrings #-} {-# LANGUAGE QuasiQuotes #-} -- | -- -- OAuth2 plugin for http://eveonline.com -- -- * Authenticates against eveonline -- * Uses EVEs unique account-user-char-hash as credentials identifier -- * Returns charName, tokenType and expires as extras -- module Yesod.Auth.OAuth2.EveOnline ( oauth2Eve , oauth2EveWidget , oauth2EveScoped , WidgetType(..) , module Yesod.Auth.OAuth2 ) where #if __GLASGOW_HASKELL__ < 710 import Control.Applicative ((<$>), (<*>)) #endif import Control.Exception.Lifted import Control.Monad (mzero) import Data.Aeson import Data.Monoid ((<>)) import Data.Text as T (Text,unwords) import Data.ByteString as B (ByteString) import Data.Text.Encoding (encodeUtf8, decodeUtf8) import Network.HTTP.Conduit (Manager) import Yesod.Auth import Yesod.Auth.OAuth2 import Yesod.Core.Widget import qualified Data.Text as T data YesodAuth m => WidgetType m = BigWhite | SmallWhite | BigBlack | SmallBlack | Custom (WidgetT m IO ()) data EveUser = EveUser { eveUserId :: Int , eveUserName :: Text , eveUserExpire :: Text , eveScopes :: [Text] , eveTokenType :: Text , eveCharOwnerHash :: Text } instance FromJSON EveUser where parseJSON (Object o) = EveUser <$> o .: "CharacterID" <*> o .: "CharacterName" <*> o .: "ExpiresOn" <*> (T.words <$> o .: "Scopes") <*> o .: "TokenType" <*> o .: "CharacterOwnerHash" parseJSON _ = mzero oauth2Eve :: YesodAuth m => Text -- ^ Client ID -> Text -- ^ Client Secret -> AuthPlugin m oauth2Eve clientId clientSecret = oauth2EveScoped clientId clientSecret ["publicData"] Nothing oauth2EveWidget :: YesodAuth m => Text -- ^ Client ID -> Text -- ^ Client Secret -> WidgetType m -> AuthPlugin m oauth2EveWidget clientId clientSecret w = oauth2EveScoped clientId clientSecret ["publicData"] (Just . toWidget $ w) where toWidget :: YesodAuth m => WidgetType m -> WidgetT m IO () toWidget BigWhite = [whamlet||] toWidget BigBlack = [whamlet||] toWidget SmallWhite = [whamlet||] toWidget SmallBlack = [whamlet||] toWidget (Custom a) = a oauth2EveScoped :: YesodAuth m => Text -- ^ Client ID -> Text -- ^ Client Secret -> [Text] -- ^ List of scopes to request -> Maybe (WidgetT m IO ()) -- ^ Login-Widget -> AuthPlugin m oauth2EveScoped clientId clientSecret scopes widget = case widget of Just w -> authOAuth2Widget "eveonline" oauth fetchEveProfile w Nothing -> authOAuth2 "eveonline" oauth fetchEveProfile where oauth = OAuth2 { oauthClientId = encodeUtf8 clientId , oauthClientSecret = encodeUtf8 clientSecret , oauthOAuthorizeEndpoint = encodeUtf8 $ "https://login.eveonline.com/oauth/authorize?response_type=code&scope=" <> T.intercalate " " scopes , oauthAccessTokenEndpoint = "https://login.eveonline.com/oauth/token" , oauthCallback = Nothing } fetchEveProfile :: Manager -> AccessToken -> IO (Creds m) fetchEveProfile manager token = do userResult <- authGetJSON manager token "https://login.eveonline.com/oauth/verify" case userResult of Right user -> return $ toCreds user token Left err-> throwIO $ InvalidProfileResponse "eveonline" err toCreds :: EveUser -> AccessToken -> Creds m toCreds user token = Creds { credsPlugin = "eveonline" , credsIdent = T.pack $ show $ eveCharOwnerHash user , credsExtra = [ ("charName", eveUserName user) , ("tokenType", eveTokenType user) , ("expires", eveUserExpire user) ] }