module Data.Swagger.Model.Authorisation where
import Data.Aeson
import Data.Swagger.Model.Util
import Data.Text (Text)
default (Text)
data Authorisation
    = BasicAuth
    | ApiKey
        { passAs  :: PassMethod
        , keyname :: Text
        }
    | OAuth2
        { scopes     :: Maybe [Scope]
        , grantTypes :: GrantTypes
        }
    deriving Show
data PassMethod
    = PassAsHeader
    | PassAsQuery
    deriving (Eq, Show)
data Scope = Scope
    { scope       :: Text
    , description :: Maybe Text
    } deriving Show
data GrantTypes = GrantTypes
    { implicit :: Maybe ImplicitGrant
    , authCode :: Maybe AuthCode
    } deriving Show
data ImplicitGrant = ImplicitGrant
    { loginEndpoint :: LoginEndpoint
    , tokenName     :: Maybe Text
    } deriving Show
newtype LoginEndpoint = LoginEndpoint
    { loginUrl :: Text
    } deriving Show
data AuthCode = AuthCode
    { tokenRequestEndpoint :: TokenRequestEndpoint
    , tokenEndpoint        :: TokenEndpoint
    } deriving Show
data TokenRequestEndpoint = TokenRequestEndpoint
    { tokenRequestUrl  :: Text
    , clientIdName     :: Maybe Text
    , clientSecretName :: Maybe Text
    } deriving Show
data TokenEndpoint = TokenEndpoint
    { tokenEndpointUrl       :: Text
    , tokenEndpointTokenName :: Maybe Text
    } deriving Show
instance ToJSON Authorisation where
    toJSON BasicAuth    = object [ "type" .= "basicAuth" ]
    toJSON (ApiKey p k) = object
        $ "type"    .= "apiKey"
        # "passAs"  .= p
        # "keyname" .= k
        # []
    toJSON (OAuth2 s g) = object
        $ "type"       .= "oauth2"
        # "scopes"     .= s
        # "grantTypes" .= g
        # []
instance ToJSON PassMethod where
    toJSON PassAsHeader = "header"
    toJSON PassAsQuery  = "query"
instance ToJSON Scope where
    toJSON a = object
        $ "scope"       .= scope a
        # "description" .= description a
        # []
instance ToJSON GrantTypes where
    toJSON a = object
        $ "implicit"           .= implicit a
        # "authorization_code" .= authCode a
        # []
instance ToJSON ImplicitGrant where
    toJSON a = object
        $ "loginEndpoint" .= loginEndpoint a
        # "tokenName"     .= tokenName a
        # []
instance ToJSON AuthCode where
    toJSON a = object
        $ "tokenRequestEndpoint" .= tokenRequestEndpoint a
        # "tokenEndpoint"        .= tokenEndpoint a
        # []
instance ToJSON LoginEndpoint where
    toJSON a = object [ "url" .= loginUrl a ]
instance ToJSON TokenRequestEndpoint where
    toJSON a = object
        $ "url"              .= tokenRequestUrl a
        # "clientIdName"     .= clientIdName a
        # "clientSecretName" .= clientSecretName a
        # []
instance ToJSON TokenEndpoint where
    toJSON a = object
        $ "url"       .= tokenEndpointUrl a
        # "tokenName" .= tokenEndpointTokenName a
        # []