From b712fcdec341bb3b07a95fbcf5e77c6794f7da01 Mon Sep 17 00:00:00 2001 From: KtorZ Date: Sat, 15 Feb 2020 14:26:58 +0100 Subject: [PATCH] rework examples - Add auth0 example with authorization code and PKCE support - Add facebook example - Make them more readable and avoid unrelated code in examples - Add README to summarize information --- examples/Makefile | 24 + examples/assets/css/branding.css | 19 + examples/assets/css/style.css | 110 +++ examples/authorization-code/Main.elm | 224 ------ examples/authorization-code/index.html | 26 - examples/common/OAuth/Examples/Common.elm | 397 ----------- examples/dist/.gitkeep | 0 examples/elm.json | 30 - examples/implicit/Main.elm | 176 ----- examples/implicit/index.html | 11 - examples/providers/auth0/README.md | 33 + .../auth0/authorization-code/Main.elm | 613 ++++++++++++++++ .../auth0/authorization-code/README.md | 5 + .../auth0/authorization-code/elm.json | 34 + .../providers/auth0/authorization-code/src | 1 + examples/providers/auth0/implicit/Main.elm | 513 ++++++++++++++ examples/providers/auth0/implicit/README.md | 5 + examples/providers/auth0/implicit/elm.json | 34 + examples/providers/auth0/implicit/src | 1 + examples/providers/auth0/pkce/Main.elm | 655 ++++++++++++++++++ examples/providers/auth0/pkce/README.md | 5 + examples/providers/auth0/pkce/elm.json | 34 + examples/providers/auth0/pkce/src | 1 + examples/providers/facebook/README.md | 27 + examples/providers/facebook/implicit/Main.elm | 558 +++++++++++++++ examples/providers/facebook/implicit/elm.json | 34 + examples/providers/facebook/implicit/src | 1 + examples/providers/google/README.md | 27 + examples/providers/google/implicit/Main.elm | 513 ++++++++++++++ examples/providers/google/implicit/README.md | 5 + examples/providers/google/implicit/elm.json | 34 + examples/providers/google/implicit/src | 1 + examples/providers/spotify/README.md | 27 + examples/providers/spotify/implicit/Main.elm | 513 ++++++++++++++ examples/providers/spotify/implicit/README.md | 5 + examples/providers/spotify/implicit/elm.json | 34 + examples/providers/spotify/implicit/src | 1 + examples/src | 1 - guides/facebook/README.md | 62 -- guides/facebook/logo.png | Bin 11751 -> 0 bytes guides/github/README.md | 57 -- guides/github/logo.png | Bin 9924 -> 0 bytes src/OAuth/AuthorizationCode.elm | 12 +- src/OAuth/AuthorizationCode/PKCE.elm | 7 + 44 files changed, 3883 insertions(+), 987 deletions(-) create mode 100644 examples/Makefile create mode 100644 examples/assets/css/branding.css create mode 100644 examples/assets/css/style.css delete mode 100644 examples/authorization-code/Main.elm delete mode 100644 examples/authorization-code/index.html delete mode 100644 examples/common/OAuth/Examples/Common.elm create mode 100644 examples/dist/.gitkeep delete mode 100644 examples/elm.json delete mode 100644 examples/implicit/Main.elm delete mode 100644 examples/implicit/index.html create mode 100644 examples/providers/auth0/README.md create mode 100644 examples/providers/auth0/authorization-code/Main.elm create mode 100644 examples/providers/auth0/authorization-code/README.md create mode 100644 examples/providers/auth0/authorization-code/elm.json create mode 120000 examples/providers/auth0/authorization-code/src create mode 100644 examples/providers/auth0/implicit/Main.elm create mode 100644 examples/providers/auth0/implicit/README.md create mode 100644 examples/providers/auth0/implicit/elm.json create mode 120000 examples/providers/auth0/implicit/src create mode 100644 examples/providers/auth0/pkce/Main.elm create mode 100644 examples/providers/auth0/pkce/README.md create mode 100644 examples/providers/auth0/pkce/elm.json create mode 120000 examples/providers/auth0/pkce/src create mode 100644 examples/providers/facebook/README.md create mode 100644 examples/providers/facebook/implicit/Main.elm create mode 100644 examples/providers/facebook/implicit/elm.json create mode 120000 examples/providers/facebook/implicit/src create mode 100644 examples/providers/google/README.md create mode 100644 examples/providers/google/implicit/Main.elm create mode 100644 examples/providers/google/implicit/README.md create mode 100644 examples/providers/google/implicit/elm.json create mode 120000 examples/providers/google/implicit/src create mode 100644 examples/providers/spotify/README.md create mode 100644 examples/providers/spotify/implicit/Main.elm create mode 100644 examples/providers/spotify/implicit/README.md create mode 100644 examples/providers/spotify/implicit/elm.json create mode 120000 examples/providers/spotify/implicit/src delete mode 120000 examples/src delete mode 100644 guides/facebook/README.md delete mode 100644 guides/facebook/logo.png delete mode 100644 guides/github/README.md delete mode 100644 guides/github/logo.png diff --git a/examples/Makefile b/examples/Makefile new file mode 100644 index 0000000..a0b5a75 --- /dev/null +++ b/examples/Makefile @@ -0,0 +1,24 @@ +ELM=elm make --optimize +DIST=dist +SRC=providers +OUTPUT=../../../$(DIST)/app.min.js + +.PHONY: help start + +help: + @echo -n "Usage: make /\n\nExamples:\n make auth0/pkce\n make google/authorization-code\n make facebook/implicit" + +start: + python -m SimpleHTTPServer + +%/implicit: $(DIST) + cd $(SRC)/$@ && $(ELM) --output=$(OUTPUT) *.elm + +%/authorization-code: $(DIST) + cd $(SRC)/$@ && $(ELM) --output=$(OUTPUT) *.elm + +%/pkce: $(DIST) + cd $(SRC)/$@ && $(ELM) --output=$(OUTPUT) *.elm + +$(DIST): + mkdir -p $@ diff --git a/examples/assets/css/branding.css b/examples/assets/css/branding.css new file mode 100644 index 0000000..9ef7d5e --- /dev/null +++ b/examples/assets/css/branding.css @@ -0,0 +1,19 @@ +.btn-auth0 { + background: 0 center no-repeat url(''); + background-size: 3em; +} + +.btn-google { + background: 0 center no-repeat url(''); + background-size: 3em; +} + +.btn-spotify { + background: 0 center no-repeat url(''); + background-size: 3em; +} + +.btn-facebook { + background: 0 center no-repeat url(''); + background-size: 3em 2.5em; +} diff --git a/examples/assets/css/style.css b/examples/assets/css/style.css new file mode 100644 index 0000000..b55d133 --- /dev/null +++ b/examples/assets/css/style.css @@ -0,0 +1,110 @@ +html, body { + margin: 0; + padding: 0; + width: 100%; + height: 100%; +} + +body { + font-family: Roboto, Arial, sans-serif; + width: 100%; +} + +button { + border: none; + box-shadow: rgba(0,0,0,0.25) 0px 2px 4px 0px; + color: #757575; + cursor: pointer; + font-size: 24px; + height: 3em; + margin: 0.5em 0; + outline: none; + padding: 0 1em 0 4em; + text-align: right; + width: 10em; +} + +button::-moz-focus-inner { + border: 0; +} + +img.avatar { + height: 15em; + width: 15em; + border-radius: 50%; + box-shadow: rgba(0,0,0,0.25) 0 0 4px 2px; +} + +img.avatar + p { + margin: 2em; + font: 24px Roboto, Arial; + color: #757575; +} + +.flex { + align-items: center; + display: flex; + justify-content: center; + width: 100%; +} + +.flex-column { + flex-direction: column; + justify-content: center; +} + +.flex-space-around { + height: 100%; + justify-content: space-around; +} + + +.step { + width: 2em; + height: 2em; + border: 1px solid; + border-radius: 50%; + border-color: #95a5a6; +} + +.step > * { + position: relative; + display: block; + top: 2.5em; + color: #95a5a6; + white-space: nowrap; + font-variant: small-caps; +} + +.step-separator { + width: 7.5em; + height: 0.1em; + background-color: #95a5a6; +} + +.step-active { + border-color: #2ecc71; + background-color: #2ecc71; + transition: all 250ms ease-in; +} + +.step-active > * { + font-weight: bold; + color: #2ecc71; +} + +.step-errored { + border-color: #e74c3c; + background-color: #e74c3c; + transition: all 250ms ease-in; +} + +.step-errored > * { + font-weight: bold; + color: #e74c3c;; +} + +.step-separator.step-active { + height: 0.2em; + transition: all 250ms ease-in; +} diff --git a/examples/authorization-code/Main.elm b/examples/authorization-code/Main.elm deleted file mode 100644 index 454b10b..0000000 --- a/examples/authorization-code/Main.elm +++ /dev/null @@ -1,224 +0,0 @@ -module Main exposing (main) - -import Browser exposing (application) -import Browser.Navigation as Navigation exposing (Key) -import Html exposing (..) -import Html.Attributes exposing (..) -import Http -import Json.Decode as Json -import OAuth -import OAuth.AuthorizationCode -import OAuth.Examples.Common exposing (..) -import Url exposing (Url) - - -main : Program { randomBytes : String } Model Msg -main = - application - { init = init - , view = - view "Elm OAuth2 Example - AuthorizationCode Flow" - { buttons = - [ viewSignInButton Google SignInRequested - , viewSignInButton Spotify SignInRequested - , viewSignInButton LinkedIn SignInRequested - ] - , sideNote = sideNote - , onSignOut = SignOutRequested - } - , update = update - , subscriptions = always Sub.none - , onUrlRequest = always NoOp - , onUrlChange = always NoOp - } - - - --- --- Msg --- - - -type - Msg - -- No Operation, terminal case - = NoOp - -- The 'sign-in' button has been hit - | SignInRequested OAuthConfiguration - -- The 'sign-out' button has been hit - | SignOutRequested - -- Got a response from the googleapis token endpoint - | GotAccessToken OAuthConfiguration (Result Http.Error OAuth.AuthorizationCode.AuthenticationSuccess) - -- Got a response from the googleapis info endpoint - | GotUserInfo (Result Http.Error Profile) - - -getUserInfo : OAuthConfiguration -> OAuth.Token -> Cmd Msg -getUserInfo { profileEndpoint, profileDecoder } token = - Http.request - { method = "GET" - , body = Http.emptyBody - , headers = OAuth.useToken token [] - , url = Url.toString profileEndpoint - , expect = Http.expectJson GotUserInfo profileDecoder - , timeout = Nothing - , tracker = Nothing - } - - -getAccessToken : OAuthConfiguration -> Url -> String -> Cmd Msg -getAccessToken ({ clientId, secret, tokenEndpoint } as config) redirectUri code = - Http.request <| - OAuth.AuthorizationCode.makeTokenRequest - (GotAccessToken config) - { credentials = - { clientId = clientId - , secret = Just secret - } - , code = code - , url = tokenEndpoint - , redirectUri = redirectUri - } - - - --- --- Init --- - - -init : { randomBytes : String } -> Url -> Key -> ( Model, Cmd Msg ) -init { randomBytes } origin _ = - let - model = - makeInitModel randomBytes origin - in - case OAuth.AuthorizationCode.parseCode origin of - OAuth.AuthorizationCode.Success { code, state } -> - if Maybe.map randomBytesFromState state /= Just model.state then - ( { model | error = Just "'state' doesn't match, the request has likely been forged by an adversary!" } - , Cmd.none - ) - - else - case Maybe.andThen (Maybe.map configurationFor << oauthProviderFromState) state of - Nothing -> - ( { model | error = Just "Couldn't recover OAuthProvider from state" } - , Cmd.none - ) - - Just config -> - ( model - , getAccessToken config model.redirectUri code - ) - - OAuth.AuthorizationCode.Empty -> - ( model, Cmd.none ) - - OAuth.AuthorizationCode.Error err -> - ( { model | error = Just (OAuth.errorCodeToString err.error) } - , Cmd.none - ) - - - --- --- Update --- - - -update : Msg -> Model -> ( Model, Cmd Msg ) -update msg model = - case msg of - NoOp -> - ( model, Cmd.none ) - - SignInRequested { scope, provider, clientId, authorizationEndpoint } -> - let - auth = - { clientId = clientId - , redirectUri = model.redirectUri - , scope = scope - , state = Just (makeState model.state provider) - , url = authorizationEndpoint - } - in - ( model - , auth |> OAuth.AuthorizationCode.makeAuthorizationUrl |> Url.toString |> Navigation.load - ) - - SignOutRequested -> - ( model - , Navigation.load (Url.toString model.redirectUri) - ) - - GotAccessToken config res -> - case res of - Err (Http.BadBody body) -> - case Json.decodeString OAuth.AuthorizationCode.defaultAuthenticationErrorDecoder body of - Ok { error, errorDescription } -> - let - errMsg = - "Unable to retrieve token: " ++ errorResponseToString { error = error, errorDescription = errorDescription } - in - ( { model | error = Just errMsg } - , Cmd.none - ) - - _ -> - ( { model | error = Just ("Unable to retrieve token: " ++ body) } - , Cmd.none - ) - - Err _ -> - ( { model | error = Just "Unable to retrieve token: HTTP request failed. CORS is likely disabled on the authorization server." } - , Cmd.none - ) - - Ok { token } -> - ( { model | token = Just token } - , getUserInfo config token - ) - - GotUserInfo res -> - case res of - Err _ -> - ( { model | error = Just "Unable to retrieve user profile: HTTP request failed." } - , Cmd.none - ) - - Ok profile -> - ( { model | profile = Just profile } - , Cmd.none - ) - - - --- --- Side Note --- - - -sideNote : List (Html msg) -sideNote = - [ h1 [] [ text "Authorization Code" ] - , p [] - [ text """ -This simple demo gives an example on how to implement the OAuth-2.0 -Authorization Code grant using Elm. Keep in mind that this example -is fully written Elm whereas you'd likely to the 'authentication' step -server-side. Actually, most well-known authorization servers don't -enable CORS on the authentication endpoint, making it impossible to perform -this operation client-side. - """ - ] - , p [] - [ text "A few interesting notes about this demo:" - , br [] [] - , ul [] - [ li [ style "margin" "0.5em 0" ] [ text "This demo application requires basic scopes from the authorization servers in order to display your name and profile picture, illustrating the demo." ] - , li [ style "margin" "0.5em 0" ] [ text "You can observe the URL in the browser navigation bar and requests made against the authorization servers!" ] - , li [ style "margin" "0.5em 0" ] [ text "None of the 'authentication' steps in this demo will work for it uses dummy secrets. Yet, it is still possible to do the 'authorization' step to retrieve an authorization code. You may try to submit this code via cURL to obtain an access token." ] - ] - ] - ] diff --git a/examples/authorization-code/index.html b/examples/authorization-code/index.html deleted file mode 100644 index 934268a..0000000 --- a/examples/authorization-code/index.html +++ /dev/null @@ -1,26 +0,0 @@ - - - - -
- - - - diff --git a/examples/common/OAuth/Examples/Common.elm b/examples/common/OAuth/Examples/Common.elm deleted file mode 100644 index f1fbe0a..0000000 --- a/examples/common/OAuth/Examples/Common.elm +++ /dev/null @@ -1,397 +0,0 @@ -module OAuth.Examples.Common exposing (Model, OAuthConfiguration, OAuthProvider(..), Profile, attrLogo, configurationFor, errorResponseToString, makeInitModel, makeState, oauthProviderFromState, oauthProviderFromString, oauthProviderToString, queryAsFragment, randomBytesFromState, stringDropLeftUntil, stringLeftUntil, view, viewBody, viewError, viewFetching, viewLogin, viewProfile, viewSignInButton) - -import Browser exposing (Document, application) -import Html exposing (..) -import Html.Attributes exposing (..) -import Html.Events exposing (..) -import Json.Decode as Json -import OAuth -import OAuth.Implicit exposing (defaultParsers) -import Url exposing (Protocol(..), Url) -import Url.Parser.Query as Query - - - --- Model - - -type alias Model = - { redirectUri : Url - , error : Maybe String - , token : Maybe OAuth.Token - , profile : Maybe Profile - , state : String - } - - -type alias Profile = - { name : String - , picture : String - } - - -type OAuthProvider - = Google - | Spotify - | LinkedIn - - -type alias OAuthConfiguration = - { provider : OAuthProvider - , authorizationEndpoint : Url - , tokenEndpoint : Url - , profileEndpoint : Url - , clientId : String - , secret : String - , scope : List String - , profileDecoder : Json.Decoder Profile - } - - -makeInitModel : String -> Url -> Model -makeInitModel bytes origin = - { redirectUri = { origin | query = Nothing, fragment = Nothing } - , error = Nothing - , token = Nothing - , profile = Nothing - , state = bytes - } - - - --- --- View --- - - -view : String -> { onSignOut : msg, buttons : List (Html msg), sideNote : List (Html msg) } -> Model -> Document msg -view title { onSignOut, buttons, sideNote } model = - let - content = - case ( model.token, model.profile ) of - ( Nothing, Nothing ) -> - viewLogin { buttons = buttons, sideNote = sideNote } - - ( Just token, Nothing ) -> - [ viewFetching ] - - ( _, Just profile ) -> - [ viewProfile onSignOut profile ] - in - { title = title - , body = [ viewBody model content ] - } - - -viewBody : Model -> List (Html msg) -> Html msg -viewBody model content = - div - [ style "display" "flex" - , style "align-items" "center" - , style "justify-content" "center" - , style "width" "100%" - , style "height" "98vh" - , style "font-family" "Roboto, Arial, sans-serif" - ] - (viewError model.error :: content) - - -viewLogin : { buttons : List (Html msg), sideNote : List (Html msg) } -> List (Html msg) -viewLogin { buttons, sideNote } = - [ div - [ style "display" "flex" - , style "align-items" "flex-end" - , style "justify-content" "center" - , style "flex-direction" "column" - ] - buttons - , div - [ style "background" "#bdc3c7" - , style "height" "10em" - , style "width" "0.1em" - , style "margin" "0 1em" - ] - [] - , div - [ style "width" "25em" - , style "padding" "1em 1em" - ] - sideNote - ] - - -viewFetching : Html msg -viewFetching = - div - [ style "color" "#757575" - , style "font" "Roboto Arial" - , style "text-align" "center" - , style "display" "flex" - , style "align-items" "center" - , style "justify-content" "center" - ] - [ text "fetching profile..." ] - - -viewProfile : msg -> Profile -> Html msg -viewProfile onSignOut profile = - div - [ style "display" "flex" - , style "flex-direction" "column" - , style "align-items" "center" - , style "justify-content" "center" - ] - [ img - [ src profile.picture - , style "height" "15em" - , style "width" "15em" - , style "border-radius" "50%" - , style "box-shadow" "rgba(0,0,0,0.25) 0 0 4px 2px" - ] - [] - , div - [ style "margin" "2em" - , style "font" "24px Roboto, Arial" - , style "color" "#757575" - ] - [ text <| profile.name ] - , div - [] - [ button - [ onClick onSignOut - , style "font-size" "24px" - , style "cursor" "pointer" - , style "height" "2em" - , style "width" "8em" - ] - [ text "Sign Out" - ] - ] - ] - - -viewError : Maybe String -> Html msg -viewError error = - case error of - Nothing -> - div [ style "display" "none" ] [] - - Just msg -> - div - [ style "width" "100%" - , style "padding" "1em" - , style "text-align" "center" - , style "background" "#e74c3c" - , style "color" "#ffffff" - , style "position" "absolute" - , style "top" "0" - , style "display" "block" - , style "box-sizing" "border-box" - ] - [ text msg ] - - -viewSignInButton : OAuthProvider -> (OAuthConfiguration -> msg) -> Html msg -viewSignInButton provider onSignIn = - button - [ attrLogo provider - , style "background-size" "3em" - , style "border" "none" - , style "box-shadow" "rgba(0,0,0,0.25) 0px 2px 4px 0px" - , style "color" "#757575" - , style "font-size" "24px" - , style "outline" "none" - , style "cursor" "pointer" - , style "height" "3em" - , style "width" "10em" - , style "text-align" "right" - , style "padding" "0 1em 0 4em" - , style "margin" "0.5em 0" - , onClick (onSignIn <| configurationFor provider) - ] - [ text "Sign in" ] - - -attrLogo : OAuthProvider -> Attribute msg -attrLogo provider = - let - data = - case provider of - Google -> - "" - - Spotify -> - "" - - LinkedIn -> - "" - in - style "background" ("0 center no-repeat url('" ++ data ++ "')") - - - --- --- Helpers --- - - -errorResponseToString : { error : OAuth.ErrorCode, errorDescription : Maybe String } -> String -errorResponseToString { error, errorDescription } = - let - code = - OAuth.errorCodeToString error - - desc = - errorDescription - |> Maybe.withDefault "" - |> String.replace "+" " " - in - code ++ ": " ++ desc - - -oauthProviderToString : OAuthProvider -> String -oauthProviderToString provider = - case provider of - Google -> - "google" - - Spotify -> - "spotify" - - LinkedIn -> - "linkedin" - - -oauthProviderFromString : String -> Maybe OAuthProvider -oauthProviderFromString str = - case str of - "google" -> - Just Google - - "spotify" -> - Just Spotify - - "linkedin" -> - Just LinkedIn - - _ -> - Nothing - - -makeState : String -> OAuthProvider -> String -makeState suffix provider = - oauthProviderToString provider ++ "." ++ suffix - - -oauthProviderFromState : String -> Maybe OAuthProvider -oauthProviderFromState str = - str - |> stringLeftUntil (\c -> c == ".") - |> oauthProviderFromString - - -randomBytesFromState : String -> String -randomBytesFromState str = - str - |> stringDropLeftUntil (\c -> c == ".") - - -stringDropLeftUntil : (String -> Bool) -> String -> String -stringDropLeftUntil predicate str = - let - ( h, q ) = - ( String.left 1 str, String.dropLeft 1 str ) - in - if q == "" || predicate h then - q - - else - stringDropLeftUntil predicate q - - -stringLeftUntil : (String -> Bool) -> String -> String -stringLeftUntil predicate str = - let - ( h, q ) = - ( String.left 1 str, String.dropLeft 1 str ) - in - if h == "" || predicate h then - "" - - else - h ++ stringLeftUntil predicate q - - -configurationFor : OAuthProvider -> OAuthConfiguration -configurationFor provider = - let - defaultHttpsUrl = - { protocol = Https - , host = "" - , path = "" - , port_ = Nothing - , query = Nothing - , fragment = Nothing - } - in - case provider of - Google -> - { provider = Google - , clientId = "909608474358-fkok86ks7e83c47aq01aiit47vsoh4s0.apps.googleusercontent.com" - , secret = "" - , authorizationEndpoint = { defaultHttpsUrl | host = "accounts.google.com", path = "/o/oauth2/v2/auth" } - , tokenEndpoint = { defaultHttpsUrl | host = "www.googleapis.com", path = "/oauth2/v4/token" } - , profileEndpoint = { defaultHttpsUrl | host = "www.googleapis.com", path = "/oauth2/v1/userinfo" } - , scope = [ "profile" ] - , profileDecoder = - Json.map2 Profile - (Json.field "name" Json.string) - (Json.field "picture" Json.string) - } - - Spotify -> - { provider = Spotify - , clientId = "391d08ef3d7a46558493cb822a991dbb" - , secret = "" - , authorizationEndpoint = { defaultHttpsUrl | host = "accounts.spotify.com", path = "/authorize" } - , tokenEndpoint = { defaultHttpsUrl | host = "accounts.spotify.com", path = "/api/token" } - , profileEndpoint = { defaultHttpsUrl | host = "api.spotify.com", path = "/v1/me" } - , scope = [] - , profileDecoder = - Json.map2 Profile - (Json.field "display_name" Json.string) - (Json.field "images" <| Json.index 0 <| Json.field "url" Json.string) - } - - LinkedIn -> - { provider = LinkedIn - , clientId = "778vrrt6dbp865" - , secret = "" - , authorizationEndpoint = { defaultHttpsUrl | host = "www.linkedin.com", path = "/oauth/v2/authorization" } - , tokenEndpoint = { defaultHttpsUrl | host = "www.linkedin.com", path = "/oauth/v2/accessToken" } - , profileEndpoint = { defaultHttpsUrl | host = "api.linkedin.com", path = "/v2/me" } - , scope = [ "r_basicprofile" ] - , profileDecoder = - Json.map2 Profile - (Json.field "firstName" Json.string) - (Json.field "pictureUrl" Json.string) - } - - - --- --- Hacks --- - - -queryAsFragment : Url -> Url -queryAsFragment url = - case url.fragment of - Just "_=_" -> - { url | fragment = url.query, query = Nothing } - - Nothing -> - { url | fragment = url.query, query = Nothing } - - _ -> - url diff --git a/examples/dist/.gitkeep b/examples/dist/.gitkeep new file mode 100644 index 0000000..e69de29 diff --git a/examples/elm.json b/examples/elm.json deleted file mode 100644 index f95708e..0000000 --- a/examples/elm.json +++ /dev/null @@ -1,30 +0,0 @@ -{ - "type": "application", - "source-directories": [ - "src", - "common" - ], - "elm-version": "0.19.0", - "dependencies": { - "direct": { - "elm/browser": "1.0.1", - "elm/bytes": "1.0.7", - "elm/core": "1.0.2", - "elm/html": "1.0.0", - "elm/http": "2.0.0", - "elm/json": "1.1.2", - "elm/url": "1.0.0", - "truqu/elm-base64": "2.0.4" - }, - "indirect": { - "elm/file": "1.0.1", - "elm/regex": "1.0.0", - "elm/time": "1.0.0", - "elm/virtual-dom": "1.0.0" - } - }, - "test-dependencies": { - "direct": {}, - "indirect": {} - } -} \ No newline at end of file diff --git a/examples/implicit/Main.elm b/examples/implicit/Main.elm deleted file mode 100644 index a02df52..0000000 --- a/examples/implicit/Main.elm +++ /dev/null @@ -1,176 +0,0 @@ -port module Main exposing (main) - -import Browser exposing (application) -import Browser.Navigation as Navigation exposing (Key) -import Html exposing (..) -import Html.Attributes exposing (..) -import Http -import Json.Decode as Json -import OAuth -import OAuth.Examples.Common exposing (..) -import OAuth.Implicit -import Url exposing (Url) - - -main : Program { randomBytes : String } Model Msg -main = - application - { init = init - , view = - view "Elm OAuth2 Example - Implicit Flow" - { buttons = - [ viewSignInButton Google SignInRequested - , viewSignInButton Spotify SignInRequested - , viewSignInButton LinkedIn SignInRequested - ] - , sideNote = sideNote - , onSignOut = SignOutRequested - } - , update = update - , subscriptions = always Sub.none - , onUrlRequest = always NoOp - , onUrlChange = always NoOp - } - - - --- --- Msg --- - - -type - Msg - -- No Operation, terminal case - = NoOp - -- The 'sign-in' button has been hit - | SignInRequested OAuthConfiguration - -- The 'sign-out' button has been hit - | SignOutRequested - -- Got a response from the googleapis user info - | GotUserInfo (Result Http.Error Profile) - - -getUserInfo : OAuthConfiguration -> OAuth.Token -> Cmd Msg -getUserInfo { profileEndpoint, profileDecoder } token = - Http.request - { method = "GET" - , body = Http.emptyBody - , headers = OAuth.useToken token [] - , tracker = Nothing - , url = Url.toString profileEndpoint - , expect = Http.expectJson GotUserInfo profileDecoder - , timeout = Nothing - } - - - --- --- Init --- - - -init : { randomBytes : String } -> Url -> Key -> ( Model, Cmd Msg ) -init { randomBytes } origin _ = - let - model = - makeInitModel randomBytes origin - in - case OAuth.Implicit.parseToken (queryAsFragment origin) of - OAuth.Implicit.Empty -> - ( model, Cmd.none ) - - OAuth.Implicit.Success { token, state } -> - if Maybe.map randomBytesFromState state /= Just model.state then - ( { model | error = Just "'state' doesn't match, the request has likely been forged by an adversary!" } - , Cmd.none - ) - - else - case Maybe.andThen (Maybe.map configurationFor << oauthProviderFromState) state of - Nothing -> - ( { model | error = Just "Couldn't recover OAuthProvider from state" } - , Cmd.none - ) - - Just config -> - ( { model | token = Just token } - , getUserInfo config token - ) - - OAuth.Implicit.Error { error, errorDescription } -> - ( { model | error = Just <| errorResponseToString { error = error, errorDescription = errorDescription } } - , Cmd.none - ) - - - --- --- Update --- - - -update : Msg -> Model -> ( Model, Cmd Msg ) -update msg model = - case msg of - NoOp -> - ( model, Cmd.none ) - - SignInRequested { clientId, authorizationEndpoint, provider, scope } -> - let - auth = - { clientId = clientId - , redirectUri = model.redirectUri - , scope = scope - , state = Just (makeState model.state provider) - , url = authorizationEndpoint - } - in - ( model - , auth |> OAuth.Implicit.makeAuthorizationUrl |> Url.toString |> Navigation.load - ) - - SignOutRequested -> - ( model - , Navigation.load (Url.toString model.redirectUri) - ) - - GotUserInfo res -> - case res of - Err err -> - ( { model | error = Just "Unable to fetch user profile ¯\\_(ツ)_/¯" } - , Cmd.none - ) - - Ok profile -> - ( { model | profile = Just profile } - , Cmd.none - ) - - - --- --- Side Note --- - - -sideNote : List (Html msg) -sideNote = - [ h1 [] [ text "Implicit Flow" ] - , p [] - [ text """ -This simple demo gives an example on how to implement the OAuth-2.0 -Implicit grant using Elm. This is the recommended way for most client -application as it doesn't expose any secret credentials to the end-user. - """ - ] - , p [] - [ text "A few interesting notes about this demo:" - , br [] [] - , ul [] - [ li [ style "margin" "0.5em 0" ] [ text "This demo application requires basic scopes from the authorization servers in order to display your name and profile picture, illustrating the demo." ] - , li [ style "margin" "0.5em 0" ] [ text "You can observe the URL in the browser navigation bar and requests made against the authorization servers!" ] - , li [ style "margin" "0.5em 0" ] [ text "The LinkedIn implemention doesn't work as LinkedIn only supports the 'Authorization Code' grant. Though, the button is still here to show an example of error path." ] - ] - ] - ] diff --git a/examples/implicit/index.html b/examples/implicit/index.html deleted file mode 100644 index aea6258..0000000 --- a/examples/implicit/index.html +++ /dev/null @@ -1,11 +0,0 @@ - - - - -
- - - - diff --git a/examples/providers/auth0/README.md b/examples/providers/auth0/README.md new file mode 100644 index 0000000..59d18ef --- /dev/null +++ b/examples/providers/auth0/README.md @@ -0,0 +1,33 @@ +# Auth0 + +## Authorization Flows + +Flow | Support | Remark | Example +--- | --- | --- | --- +Implicit | :heavy_check_mark: | \- | [live demo][implicit-demo] \| [source code][implicit-source] +Authorization Code | :heavy_check_mark: | \- | [live demo][authorization-demo] \| [source code][authorization-source] +Authorization Code w/ PKCE | :heavy_check_mark: | \- | [live demo][pkce-demo] \| [source code][pkce-source] +Password | :heavy_check_mark: | \- | N/A +Client Credentials | :heavy_check_mark: | Requires secret key (server-side) | N/A + +## OAuth Configuration + +\- | \- +--- | --- +Authorization Endpoint | my-app.domain.auth0.com/authorize +Token Endpoint | my-app.domain.auth0.com/oauth/token +User Info Endpoint | my-app.domain.auth0.com/userinfo + +--- + +:book: https://auth0.com/docs/getting-started/overview + + +[implicit-demo]: https://truqu.github.io/elm-oauth2/auth0/implicit/ +[implicit-source]: https://github.com/truqu/elm-oauth2/blob/master/examples/providers/auth0/implicit/Main.elm + +[authorization-demo]: https://truqu.github.io/elm-oauth2/auth0/authorization-code/ +[authorization-source]: https://github.com/truqu/elm-oauth2/blob/master/examples/providers/auth0/authorization-code/Main.elm + +[pkce-demo]: https://truqu.github.io/elm-oauth2/auth0/pkce/ +[pkce-source]: https://github.com/truqu/elm-oauth2/blob/master/examples/providers/auth0/pkce/Main.elm diff --git a/examples/providers/auth0/authorization-code/Main.elm b/examples/providers/auth0/authorization-code/Main.elm new file mode 100644 index 0000000..18bdc52 --- /dev/null +++ b/examples/providers/auth0/authorization-code/Main.elm @@ -0,0 +1,613 @@ +port module Main exposing (main) + +import Base64.Encode as Base64 +import Browser exposing (Document, application) +import Browser.Navigation as Navigation exposing (Key) +import Bytes exposing (Bytes) +import Bytes.Encode as Bytes +import Delay exposing (TimeUnit(..), after) +import Html exposing (..) +import Html.Attributes exposing (..) +import Html.Events exposing (..) +import Http +import Json.Decode as Json +import OAuth +import OAuth.AuthorizationCode as OAuth +import Url exposing (Protocol(..), Url) + + +main : Program (Maybe (List Int)) Model Msg +main = + application + { init = + Maybe.map convertBytes >> init + , update = + update + , subscriptions = + always <| randomBytes GotRandomBytes + , onUrlRequest = + always NoOp + , onUrlChange = + always NoOp + , view = + view + { title = "Auth0 - Flow: Authorization Code" + , btnClass = class "btn-auth0" + } + } + + +{-| OAuth configuration. + +Note that this demo also fetches basic user information with the obtained access token, +hence the user info endpoint and JSON decoder + +-} +configuration : Configuration +configuration = + { authorizationEndpoint = + { defaultHttpsUrl | host = "elm-oauth2.eu.auth0.com", path = "/authorize" } + , tokenEndpoint = + { defaultHttpsUrl | host = "elm-oauth2.eu.auth0.com", path = "/oauth/token" } + , userInfoEndpoint = + { defaultHttpsUrl | host = "elm-oauth2.eu.auth0.com", path = "/userinfo" } + , userInfoDecoder = + Json.map2 UserInfo + (Json.field "name" Json.string) + (Json.field "picture" Json.string) + , clientId = + "AbRrXEIRBPgkDrqR4RgdXuHoAd1mDetT" + , scope = + [ "openid", "profile" ] + } + + + +-- +-- Model +-- + + +type alias Model = + { redirectUri : Url + , flow : Flow + } + + +{-| This demo evolves around the following state-machine\* + + +--------+ + | Idle | + +--------+ + | + | Redirect user for authorization + | + v + +--------------+ + | Authorized | + +--------------+ + | + | Exchange authorization code for an access token + | + v + +-----------------+ + | Authenticated | + +-----------------+ + | + | Fetch user info using the access token + v + +--------+ + | Done | + +--------+ + +(\*) The 'Errored' state hasn't been represented here for simplicity. + +-} +type Flow + = Idle + | Authorized OAuth.AuthorizationCode + | Authenticated OAuth.Token + | Done UserInfo + | Errored Error + + +type Error + = ErrStateMismatch + | ErrAuthorization OAuth.AuthorizationError + | ErrAuthentication OAuth.AuthenticationError + | ErrHTTPGetAccessToken + | ErrHTTPGetUserInfo + + +type alias UserInfo = + { name : String + , picture : String + } + + +type alias Configuration = + { authorizationEndpoint : Url + , tokenEndpoint : Url + , userInfoEndpoint : Url + , userInfoDecoder : Json.Decoder UserInfo + , clientId : String + , scope : List String + } + + +{-| During the authentication flow, we'll run twice into the `init` function: + + - The first time, for the application very first run. And we proceed with the `Idle` state, + waiting for the user (a.k.a you) to request a sign in. + + - The second time, after a sign in has been requested, the user is redirected to the + authorization server and redirects the user back to our application, with a code + and other fields as query parameters. + +When query params are present (and valid), we consider the user `Authorized`. + +-} +init : Maybe { state : String } -> Url -> Key -> ( Model, Cmd Msg ) +init mflags origin navigationKey = + let + redirectUri = + { origin | query = Nothing, fragment = Nothing } + + clearUrl = + Navigation.replaceUrl navigationKey (Url.toString redirectUri) + in + case OAuth.parseCode origin of + OAuth.Empty -> + ( { flow = Idle, redirectUri = redirectUri } + , Cmd.none + ) + + -- It is important to set a `state` when making the authorization request + -- and to verify it after the redirection. The state can be anything but its primary + -- usage is to prevent cross-site request forgery; at minima, it should be a short, + -- non-guessable string, generated on the fly. + -- + -- We remember any previously generated state state using the browser's local storage + -- and give it back (if present) to the elm application upon start + OAuth.Success { code, state } -> + case mflags of + Nothing -> + ( { flow = Errored ErrStateMismatch, redirectUri = redirectUri } + , clearUrl + ) + + Just flags -> + if state /= Just flags.state then + ( { flow = Errored ErrStateMismatch, redirectUri = redirectUri } + , clearUrl + ) + + else + ( { flow = Authorized code, redirectUri = redirectUri } + , Cmd.batch + -- Artificial delay to make the live demo easier to follow. + -- In practice, the access token could be requested right here. + [ after 750 Millisecond AccessTokenRequested + , clearUrl + ] + ) + + OAuth.Error error -> + ( { flow = Errored <| ErrAuthorization error, redirectUri = redirectUri } + , clearUrl + ) + + + +-- +-- Msg +-- + + +type Msg + = NoOp + | SignInRequested + | GotRandomBytes (List Int) + | AccessTokenRequested + | GotAccessToken (Result Http.Error OAuth.AuthenticationSuccess) + | UserInfoRequested + | GotUserInfo (Result Http.Error UserInfo) + | SignOutRequested + + +getAccessToken : Configuration -> Url -> OAuth.AuthorizationCode -> Cmd Msg +getAccessToken { clientId, tokenEndpoint } redirectUri code = + Http.request <| + OAuth.makeTokenRequest GotAccessToken + { credentials = + { clientId = clientId + , secret = Nothing + } + , code = code + , url = tokenEndpoint + , redirectUri = redirectUri + } + + +getUserInfo : Configuration -> OAuth.Token -> Cmd Msg +getUserInfo { userInfoDecoder, userInfoEndpoint } token = + Http.request + { method = "GET" + , body = Http.emptyBody + , headers = OAuth.useToken token [] + , url = Url.toString userInfoEndpoint + , expect = Http.expectJson GotUserInfo userInfoDecoder + , timeout = Nothing + , tracker = Nothing + } + + + +{- On the JavaScript's side, we have: + + app.ports.genRandomBytes.subscribe(n => { + const buffer = new Uint8Array(n); + crypto.getRandomValues(buffer); + const bytes = Array.from(buffer); + localStorage.setItem("bytes", bytes); + app.ports.randomBytes.send(bytes); + }); +-} + + +port genRandomBytes : Int -> Cmd msg + + +port randomBytes : (List Int -> msg) -> Sub msg + + + +-- +-- Update +-- + + +update : Msg -> Model -> ( Model, Cmd Msg ) +update msg model = + case ( model.flow, msg ) of + ( Idle, SignInRequested ) -> + signInRequested model + + ( Idle, GotRandomBytes bytes ) -> + gotRandomBytes model bytes + + ( Authorized code, AccessTokenRequested ) -> + accessTokenRequested model code + + ( Authorized _, GotAccessToken authenticationResponse ) -> + gotAccessToken model authenticationResponse + + ( Authenticated token, UserInfoRequested ) -> + userInfoRequested model token + + ( Authenticated _, GotUserInfo userInfoResponse ) -> + gotUserInfo model userInfoResponse + + ( Done _, SignOutRequested ) -> + signOutRequested model + + _ -> + noOp model + + +noOp : Model -> ( Model, Cmd Msg ) +noOp model = + ( model, Cmd.none ) + + +signInRequested : Model -> ( Model, Cmd Msg ) +signInRequested model = + ( { model | flow = Idle } + , genRandomBytes 16 + ) + + +gotRandomBytes : Model -> List Int -> ( Model, Cmd Msg ) +gotRandomBytes model bytes = + let + { state } = + convertBytes bytes + + authorization = + { clientId = configuration.clientId + , redirectUri = model.redirectUri + , scope = configuration.scope + , state = Just state + , url = configuration.authorizationEndpoint + } + in + ( { model | flow = Idle } + , authorization + |> OAuth.makeAuthorizationUrl + |> Url.toString + |> Navigation.load + ) + + +accessTokenRequested : Model -> OAuth.AuthorizationCode -> ( Model, Cmd Msg ) +accessTokenRequested model code = + ( { model | flow = Authorized code } + , getAccessToken configuration model.redirectUri code + ) + + +gotAccessToken : Model -> Result Http.Error OAuth.AuthenticationSuccess -> ( Model, Cmd Msg ) +gotAccessToken model authenticationResponse = + case authenticationResponse of + Err (Http.BadBody body) -> + case Json.decodeString OAuth.defaultAuthenticationErrorDecoder body of + Ok error -> + ( { model | flow = Errored <| ErrAuthentication error } + , Cmd.none + ) + + _ -> + ( { model | flow = Errored ErrHTTPGetAccessToken } + , Cmd.none + ) + + Err _ -> + ( { model | flow = Errored ErrHTTPGetAccessToken } + , Cmd.none + ) + + Ok { token } -> + ( { model | flow = Authenticated token } + , after 750 Millisecond UserInfoRequested + ) + + +userInfoRequested : Model -> OAuth.Token -> ( Model, Cmd Msg ) +userInfoRequested model token = + ( { model | flow = Authenticated token } + , getUserInfo configuration token + ) + + +gotUserInfo : Model -> Result Http.Error UserInfo -> ( Model, Cmd Msg ) +gotUserInfo model userInfoResponse = + case userInfoResponse of + Err _ -> + ( { model | flow = Errored ErrHTTPGetUserInfo } + , Cmd.none + ) + + Ok userInfo -> + ( { model | flow = Done userInfo } + , Cmd.none + ) + + +signOutRequested : Model -> ( Model, Cmd Msg ) +signOutRequested model = + ( { model | flow = Idle } + , Navigation.load (Url.toString model.redirectUri) + ) + + + +-- +-- View +-- + + +type alias ViewConfiguration msg = + { title : String + , btnClass : Attribute msg + } + + +view : ViewConfiguration Msg -> Model -> Document Msg +view ({ title } as config) model = + { title = title + , body = viewBody config model + } + + +viewBody : ViewConfiguration Msg -> Model -> List (Html Msg) +viewBody config model = + [ div [ class "flex", class "flex-column", class "flex-space-around" ] <| + case model.flow of + Idle -> + div [ class "flex" ] + [ viewAuthorizationStep False + , viewStepSeparator False + , viewAuthenticationStep False + , viewStepSeparator False + , viewGetUserInfoStep False + ] + :: viewIdle config + + Authorized _ -> + div [ class "flex" ] + [ viewAuthorizationStep True + , viewStepSeparator True + , viewAuthenticationStep False + , viewStepSeparator False + , viewGetUserInfoStep False + ] + :: viewAuthorized + + Authenticated _ -> + div [ class "flex" ] + [ viewAuthorizationStep True + , viewStepSeparator True + , viewAuthenticationStep True + , viewStepSeparator True + , viewGetUserInfoStep False + ] + :: viewAuthenticated + + Done userInfo -> + div [ class "flex" ] + [ viewAuthorizationStep True + , viewStepSeparator True + , viewAuthenticationStep True + , viewStepSeparator True + , viewGetUserInfoStep True + ] + :: viewUserInfo config userInfo + + Errored err -> + div [ class "flex" ] + [ viewErroredStep + ] + :: viewErrored err + ] + + +viewIdle : ViewConfiguration Msg -> List (Html Msg) +viewIdle { btnClass } = + [ button + [ onClick SignInRequested, btnClass ] + [ text "Sign in" ] + ] + + +viewAuthorized : List (Html Msg) +viewAuthorized = + [ span [] [ text "Authenticating..." ] + ] + + +viewAuthenticated : List (Html Msg) +viewAuthenticated = + [ span [] [ text "Getting user info..." ] + ] + + +viewUserInfo : ViewConfiguration Msg -> UserInfo -> List (Html Msg) +viewUserInfo { btnClass } { name, picture } = + [ div [ class "flex", class "flex-column" ] + [ img [ class "avatar", src picture ] [] + , p [] [ text name ] + , div [] + [ button + [ onClick SignOutRequested, btnClass ] + [ text "Sign out" ] + ] + ] + ] + + +viewErrored : Error -> List (Html Msg) +viewErrored error = + [ span [ class "span-error" ] [ viewError error ] ] + + +viewError : Error -> Html Msg +viewError e = + text <| + case e of + ErrStateMismatch -> + "'state' doesn't match, the request has likely been forged by an adversary!" + + ErrAuthorization error -> + oauthErrorToString { error = error.error, errorDescription = error.errorDescription } + + ErrAuthentication error -> + oauthErrorToString { error = error.error, errorDescription = error.errorDescription } + + ErrHTTPGetAccessToken -> + "Unable to retrieve token: HTTP request failed. CORS is likely disabled on the authorization server." + + ErrHTTPGetUserInfo -> + "Unable to retrieve user info: HTTP request failed." + + +viewAuthorizationStep : Bool -> Html Msg +viewAuthorizationStep isActive = + viewStep isActive ( "Authorization", style "left" "-110%" ) + + +viewAuthenticationStep : Bool -> Html Msg +viewAuthenticationStep isActive = + viewStep isActive ( "Authentication", style "left" "-125%" ) + + +viewGetUserInfoStep : Bool -> Html Msg +viewGetUserInfoStep isActive = + viewStep isActive ( "Get User Info", style "left" "-135%" ) + + +viewErroredStep : Html Msg +viewErroredStep = + div + [ class "step", class "step-errored" ] + [ span [ style "left" "-50%" ] [ text "Errored" ] ] + + +viewStep : Bool -> ( String, Attribute Msg ) -> Html Msg +viewStep isActive ( step, position ) = + let + stepClass = + class "step" + :: (if isActive then + [ class "step-active" ] + + else + [] + ) + in + div stepClass [ span [ position ] [ text step ] ] + + +viewStepSeparator : Bool -> Html Msg +viewStepSeparator isActive = + let + stepClass = + class "step-separator" + :: (if isActive then + [ class "step-active" ] + + else + [] + ) + in + span stepClass [] + + + +-- +-- Helpers +-- + + +toBytes : List Int -> Bytes +toBytes = + List.map Bytes.unsignedInt8 >> Bytes.sequence >> Bytes.encode + + +base64 : Bytes -> String +base64 = + Base64.bytes >> Base64.encode + + +convertBytes : List Int -> { state : String } +convertBytes = + toBytes >> base64 >> (\state -> { state = state }) + + +oauthErrorToString : { error : OAuth.ErrorCode, errorDescription : Maybe String } -> String +oauthErrorToString { error, errorDescription } = + let + desc = + errorDescription |> Maybe.withDefault "" |> String.replace "+" " " + in + OAuth.errorCodeToString error ++ ": " ++ desc + + +defaultHttpsUrl : Url +defaultHttpsUrl = + { protocol = Https + , host = "" + , path = "" + , port_ = Nothing + , query = Nothing + , fragment = Nothing + } diff --git a/examples/providers/auth0/authorization-code/README.md b/examples/providers/auth0/authorization-code/README.md new file mode 100644 index 0000000..e2e07a7 --- /dev/null +++ b/examples/providers/auth0/authorization-code/README.md @@ -0,0 +1,5 @@ +# Auth0 + +

+ +

diff --git a/examples/providers/auth0/authorization-code/elm.json b/examples/providers/auth0/authorization-code/elm.json new file mode 100644 index 0000000..a933f27 --- /dev/null +++ b/examples/providers/auth0/authorization-code/elm.json @@ -0,0 +1,34 @@ +{ + "type": "application", + "source-directories": [ + "src", + "." + ], + "elm-version": "0.19.1", + "dependencies": { + "direct": { + "andrewMacmurray/elm-delay": "3.0.0", + "elm/browser": "1.0.2", + "elm/bytes": "1.0.8", + "elm/core": "1.0.4", + "elm/html": "1.0.0", + "elm/http": "2.0.0", + "elm/json": "1.1.3", + "elm/url": "1.0.0", + "folkertdev/elm-sha2": "1.0.0", + "ivadzy/bbase64": "1.1.1" + }, + "indirect": { + "danfishgold/base64-bytes": "1.0.3", + "elm/file": "1.0.5", + "elm/regex": "1.0.0", + "elm/time": "1.0.0", + "elm/virtual-dom": "1.0.2", + "rtfeldman/elm-hex": "1.0.0" + } + }, + "test-dependencies": { + "direct": {}, + "indirect": {} + } +} diff --git a/examples/providers/auth0/authorization-code/src b/examples/providers/auth0/authorization-code/src new file mode 120000 index 0000000..dd81945 --- /dev/null +++ b/examples/providers/auth0/authorization-code/src @@ -0,0 +1 @@ +../../../../src/ \ No newline at end of file diff --git a/examples/providers/auth0/implicit/Main.elm b/examples/providers/auth0/implicit/Main.elm new file mode 100644 index 0000000..ca2d62f --- /dev/null +++ b/examples/providers/auth0/implicit/Main.elm @@ -0,0 +1,513 @@ +port module Main exposing (main) + +import Base64.Encode as Base64 +import Browser exposing (Document, application) +import Browser.Navigation as Navigation exposing (Key) +import Bytes exposing (Bytes) +import Bytes.Encode as Bytes +import Delay exposing (TimeUnit(..), after) +import Html exposing (..) +import Html.Attributes exposing (..) +import Html.Events exposing (..) +import Http +import Json.Decode as Json +import OAuth +import OAuth.Implicit as OAuth +import Url exposing (Protocol(..), Url) + + +main : Program (Maybe (List Int)) Model Msg +main = + application + { init = + Maybe.map convertBytes >> init + , update = + update + , subscriptions = + always <| randomBytes GotRandomBytes + , onUrlRequest = + always NoOp + , onUrlChange = + always NoOp + , view = + view + { title = "Auth0 - Flow: Implicit" + , btnClass = class "btn-auth0" + } + } + + +{-| OAuth configuration. + +Note that this demo also fetches basic user information with the obtained access token, +hence the user info endpoint and JSON decoder + +-} +configuration : Configuration +configuration = + { authorizationEndpoint = + { defaultHttpsUrl | host = "elm-oauth2.eu.auth0.com", path = "/authorize" } + , userInfoEndpoint = + { defaultHttpsUrl | host = "elm-oauth2.eu.auth0.com", path = "/userinfo" } + , userInfoDecoder = + Json.map2 UserInfo + (Json.field "name" Json.string) + (Json.field "picture" Json.string) + , clientId = + "AbRrXEIRBPgkDrqR4RgdXuHoAd1mDetT" + , scope = + [ "openid", "profile" ] + } + + + +-- +-- Model +-- + + +type alias Model = + { redirectUri : Url + , flow : Flow + } + + +{-| This demo evolves around the following state-machine\* + + +--------+ + | Idle | + +--------+ + | + | Redirect user for authorization + | + v + +--------------+ + | Authorized | w/ Access Token + +--------------+ + | + | Fetch user info using the access token + v + +--------+ + | Done | + +--------+ + +(\*) The 'Errored' state hasn't been represented here for simplicity. + +-} +type Flow + = Idle + | Authorized OAuth.Token + | Done UserInfo + | Errored Error + + +type Error + = ErrStateMismatch + | ErrAuthorization OAuth.AuthorizationError + | ErrHTTPGetUserInfo + + +type alias UserInfo = + { name : String + , picture : String + } + + +type alias Configuration = + { authorizationEndpoint : Url + , userInfoEndpoint : Url + , userInfoDecoder : Json.Decoder UserInfo + , clientId : String + , scope : List String + } + + +{-| During the authentication flow, we'll run twice into the `init` function: + + - The first time, for the application very first run. And we proceed with the `Idle` state, + waiting for the user (a.k.a you) to request a sign in. + + - The second time, after a sign in has been requested, the user is redirected to the + authorization server and redirects the user back to our application, with an access + token and other fields as query parameters. + +When query params are present (and valid), we consider the user `Authorized`. + +-} +init : Maybe { state : String } -> Url -> Key -> ( Model, Cmd Msg ) +init mflags origin navigationKey = + let + redirectUri = + { origin | query = Nothing, fragment = Nothing } + + clearUrl = + Navigation.replaceUrl navigationKey (Url.toString redirectUri) + in + case OAuth.parseToken origin of + OAuth.Empty -> + ( { flow = Idle, redirectUri = redirectUri } + , Cmd.none + ) + + -- It is important to set a `state` when making the authorization request + -- and to verify it after the redirection. The state can be anything but its primary + -- usage is to prevent cross-site request forgery; at minima, it should be a short, + -- non-guessable string, generated on the fly. + -- + -- We remember any previously generated state state using the browser's local storage + -- and give it back (if present) to the elm application upon start + OAuth.Success { token, state } -> + case mflags of + Nothing -> + ( { flow = Errored ErrStateMismatch, redirectUri = redirectUri } + , clearUrl + ) + + Just flags -> + if state /= Just flags.state then + ( { flow = Errored ErrStateMismatch, redirectUri = redirectUri } + , clearUrl + ) + + else + ( { flow = Authorized token, redirectUri = redirectUri } + , Cmd.batch + -- Artificial delay to make the live demo easier to follow. + -- In practice, the access token could be requested right here. + [ after 750 Millisecond UserInfoRequested + , clearUrl + ] + ) + + OAuth.Error error -> + ( { flow = Errored <| ErrAuthorization error, redirectUri = redirectUri } + , clearUrl + ) + + + +-- +-- Msg +-- + + +type Msg + = NoOp + | SignInRequested + | GotRandomBytes (List Int) + | GotAccessToken (Result Http.Error OAuth.AuthorizationSuccess) + | UserInfoRequested + | GotUserInfo (Result Http.Error UserInfo) + | SignOutRequested + + +getUserInfo : Configuration -> OAuth.Token -> Cmd Msg +getUserInfo { userInfoDecoder, userInfoEndpoint } token = + Http.request + { method = "GET" + , body = Http.emptyBody + , headers = OAuth.useToken token [] + , url = Url.toString userInfoEndpoint + , expect = Http.expectJson GotUserInfo userInfoDecoder + , timeout = Nothing + , tracker = Nothing + } + + + +{- On the JavaScript's side, we have: + + app.ports.genRandomBytes.subscribe(n => { + const buffer = new Uint8Array(n); + crypto.getRandomValues(buffer); + const bytes = Array.from(buffer); + localStorage.setItem("bytes", bytes); + app.ports.randomBytes.send(bytes); + }); +-} + + +port genRandomBytes : Int -> Cmd msg + + +port randomBytes : (List Int -> msg) -> Sub msg + + + +-- +-- Update +-- + + +update : Msg -> Model -> ( Model, Cmd Msg ) +update msg model = + case ( model.flow, msg ) of + ( Idle, SignInRequested ) -> + signInRequested model + + ( Idle, GotRandomBytes bytes ) -> + gotRandomBytes model bytes + + ( Authorized token, UserInfoRequested ) -> + userInfoRequested model token + + ( Authorized _, GotUserInfo userInfoResponse ) -> + gotUserInfo model userInfoResponse + + ( Done _, SignOutRequested ) -> + signOutRequested model + + _ -> + noOp model + + +noOp : Model -> ( Model, Cmd Msg ) +noOp model = + ( model, Cmd.none ) + + +signInRequested : Model -> ( Model, Cmd Msg ) +signInRequested model = + ( { model | flow = Idle } + , genRandomBytes 16 + ) + + +gotRandomBytes : Model -> List Int -> ( Model, Cmd Msg ) +gotRandomBytes model bytes = + let + { state } = + convertBytes bytes + + authorization = + { clientId = configuration.clientId + , redirectUri = model.redirectUri + , scope = configuration.scope + , state = Just state + , url = configuration.authorizationEndpoint + } + in + ( { model | flow = Idle } + , authorization + |> OAuth.makeAuthorizationUrl + |> Url.toString + |> Navigation.load + ) + + +userInfoRequested : Model -> OAuth.Token -> ( Model, Cmd Msg ) +userInfoRequested model token = + ( { model | flow = Authorized token } + , getUserInfo configuration token + ) + + +gotUserInfo : Model -> Result Http.Error UserInfo -> ( Model, Cmd Msg ) +gotUserInfo model userInfoResponse = + case userInfoResponse of + Err _ -> + ( { model | flow = Errored ErrHTTPGetUserInfo } + , Cmd.none + ) + + Ok userInfo -> + ( { model | flow = Done userInfo } + , Cmd.none + ) + + +signOutRequested : Model -> ( Model, Cmd Msg ) +signOutRequested model = + ( { model | flow = Idle } + , Navigation.load (Url.toString model.redirectUri) + ) + + + +-- +-- View +-- + + +type alias ViewConfiguration msg = + { title : String + , btnClass : Attribute msg + } + + +view : ViewConfiguration Msg -> Model -> Document Msg +view ({ title } as config) model = + { title = title + , body = viewBody config model + } + + +viewBody : ViewConfiguration Msg -> Model -> List (Html Msg) +viewBody config model = + [ div [ class "flex", class "flex-column", class "flex-space-around" ] <| + case model.flow of + Idle -> + div [ class "flex" ] + [ viewAuthorizationStep False + , viewStepSeparator False + , viewGetUserInfoStep False + ] + :: viewIdle config + + Authorized _ -> + div [ class "flex" ] + [ viewAuthorizationStep True + , viewStepSeparator True + , viewGetUserInfoStep False + ] + :: viewAuthorized + + Done userInfo -> + div [ class "flex" ] + [ viewAuthorizationStep True + , viewStepSeparator True + , viewGetUserInfoStep True + ] + :: viewUserInfo config userInfo + + Errored err -> + div [ class "flex" ] + [ viewErroredStep + ] + :: viewErrored err + ] + + +viewIdle : ViewConfiguration Msg -> List (Html Msg) +viewIdle { btnClass } = + [ button + [ onClick SignInRequested, btnClass ] + [ text "Sign in" ] + ] + + +viewAuthorized : List (Html Msg) +viewAuthorized = + [ span [] [ text "Getting user info..." ] + ] + + +viewUserInfo : ViewConfiguration Msg -> UserInfo -> List (Html Msg) +viewUserInfo { btnClass } { name, picture } = + [ div [ class "flex", class "flex-column" ] + [ img [ class "avatar", src picture ] [] + , p [] [ text name ] + , div [] + [ button + [ onClick SignOutRequested, btnClass ] + [ text "Sign out" ] + ] + ] + ] + + +viewErrored : Error -> List (Html Msg) +viewErrored error = + [ span [ class "span-error" ] [ viewError error ] ] + + +viewError : Error -> Html Msg +viewError e = + text <| + case e of + ErrStateMismatch -> + "'state' doesn't match, the request has likely been forged by an adversary!" + + ErrAuthorization error -> + oauthErrorToString { error = error.error, errorDescription = error.errorDescription } + + ErrHTTPGetUserInfo -> + "Unable to retrieve user info: HTTP request failed." + + +viewAuthorizationStep : Bool -> Html Msg +viewAuthorizationStep isActive = + viewStep isActive ( "Authorization", style "left" "-110%" ) + + +viewGetUserInfoStep : Bool -> Html Msg +viewGetUserInfoStep isActive = + viewStep isActive ( "Get User Info", style "left" "-135%" ) + + +viewErroredStep : Html Msg +viewErroredStep = + div + [ class "step", class "step-errored" ] + [ span [ style "left" "-50%" ] [ text "Errored" ] ] + + +viewStep : Bool -> ( String, Attribute Msg ) -> Html Msg +viewStep isActive ( step, position ) = + let + stepClass = + class "step" + :: (if isActive then + [ class "step-active" ] + + else + [] + ) + in + div stepClass [ span [ position ] [ text step ] ] + + +viewStepSeparator : Bool -> Html Msg +viewStepSeparator isActive = + let + stepClass = + class "step-separator" + :: (if isActive then + [ class "step-active" ] + + else + [] + ) + in + span stepClass [] + + + +-- +-- Helpers +-- + + +toBytes : List Int -> Bytes +toBytes = + List.map Bytes.unsignedInt8 >> Bytes.sequence >> Bytes.encode + + +base64 : Bytes -> String +base64 = + Base64.bytes >> Base64.encode + + +convertBytes : List Int -> { state : String } +convertBytes = + toBytes >> base64 >> (\state -> { state = state }) + + +oauthErrorToString : { error : OAuth.ErrorCode, errorDescription : Maybe String } -> String +oauthErrorToString { error, errorDescription } = + let + desc = + errorDescription |> Maybe.withDefault "" |> String.replace "+" " " + in + OAuth.errorCodeToString error ++ ": " ++ desc + + +defaultHttpsUrl : Url +defaultHttpsUrl = + { protocol = Https + , host = "" + , path = "" + , port_ = Nothing + , query = Nothing + , fragment = Nothing + } diff --git a/examples/providers/auth0/implicit/README.md b/examples/providers/auth0/implicit/README.md new file mode 100644 index 0000000..e2e07a7 --- /dev/null +++ b/examples/providers/auth0/implicit/README.md @@ -0,0 +1,5 @@ +# Auth0 + +

+ +

diff --git a/examples/providers/auth0/implicit/elm.json b/examples/providers/auth0/implicit/elm.json new file mode 100644 index 0000000..a933f27 --- /dev/null +++ b/examples/providers/auth0/implicit/elm.json @@ -0,0 +1,34 @@ +{ + "type": "application", + "source-directories": [ + "src", + "." + ], + "elm-version": "0.19.1", + "dependencies": { + "direct": { + "andrewMacmurray/elm-delay": "3.0.0", + "elm/browser": "1.0.2", + "elm/bytes": "1.0.8", + "elm/core": "1.0.4", + "elm/html": "1.0.0", + "elm/http": "2.0.0", + "elm/json": "1.1.3", + "elm/url": "1.0.0", + "folkertdev/elm-sha2": "1.0.0", + "ivadzy/bbase64": "1.1.1" + }, + "indirect": { + "danfishgold/base64-bytes": "1.0.3", + "elm/file": "1.0.5", + "elm/regex": "1.0.0", + "elm/time": "1.0.0", + "elm/virtual-dom": "1.0.2", + "rtfeldman/elm-hex": "1.0.0" + } + }, + "test-dependencies": { + "direct": {}, + "indirect": {} + } +} diff --git a/examples/providers/auth0/implicit/src b/examples/providers/auth0/implicit/src new file mode 120000 index 0000000..dd81945 --- /dev/null +++ b/examples/providers/auth0/implicit/src @@ -0,0 +1 @@ +../../../../src/ \ No newline at end of file diff --git a/examples/providers/auth0/pkce/Main.elm b/examples/providers/auth0/pkce/Main.elm new file mode 100644 index 0000000..c955ddc --- /dev/null +++ b/examples/providers/auth0/pkce/Main.elm @@ -0,0 +1,655 @@ +port module Main exposing (main) + +import Base64.Encode as Base64 +import Browser exposing (Document, application) +import Browser.Navigation as Navigation exposing (Key) +import Bytes exposing (Bytes) +import Bytes.Encode as Bytes +import Delay exposing (TimeUnit(..), after) +import Html exposing (..) +import Html.Attributes exposing (..) +import Html.Events exposing (..) +import Http +import Json.Decode as Json +import OAuth +import OAuth.AuthorizationCode.PKCE as OAuth +import Url exposing (Protocol(..), Url) + + +main : Program (Maybe (List Int)) Model Msg +main = + application + { init = + Maybe.andThen convertBytes >> init + , update = + update + , subscriptions = + always <| randomBytes GotRandomBytes + , onUrlRequest = + always NoOp + , onUrlChange = + always NoOp + , view = + view + { title = "Auth0 - Flow: Authorization Code w/ PKCE" + , btnClass = class "btn-auth0" + } + } + + +{-| OAuth configuration. + +Note that this demo also fetches basic user information with the obtained access token, +hence the user info endpoint and JSON decoder + +-} +configuration : Configuration +configuration = + { authorizationEndpoint = + { defaultHttpsUrl | host = "elm-oauth2.eu.auth0.com", path = "/authorize" } + , tokenEndpoint = + { defaultHttpsUrl | host = "elm-oauth2.eu.auth0.com", path = "/oauth/token" } + , userInfoEndpoint = + { defaultHttpsUrl | host = "elm-oauth2.eu.auth0.com", path = "/userinfo" } + , userInfoDecoder = + Json.map2 UserInfo + (Json.field "name" Json.string) + (Json.field "picture" Json.string) + , clientId = + "AbRrXEIRBPgkDrqR4RgdXuHoAd1mDetT" + , scope = + [ "openid", "profile" ] + } + + + +-- +-- Model +-- + + +type alias Model = + { redirectUri : Url + , flow : Flow + } + + +{-| This demo evolves around the following state-machine\* + + +--------+ + | Idle | + +--------+ + | + | Redirect user for authorization + | + v + +--------------+ + | Authorized | + +--------------+ + | + | Exchange authorization code for an access token + | + v + +-----------------+ + | Authenticated | + +-----------------+ + | + | Fetch user info using the access token + v + +--------+ + | Done | + +--------+ + +(\*) The 'Errored' state hasn't been represented here for simplicity. + +-} +type Flow + = Idle + | Authorized OAuth.AuthorizationCode OAuth.CodeVerifier + | Authenticated OAuth.Token + | Done UserInfo + | Errored Error + + +type Error + = ErrStateMismatch + | ErrFailedToConvertBytes + | ErrAuthorization OAuth.AuthorizationError + | ErrAuthentication OAuth.AuthenticationError + | ErrHTTPGetAccessToken + | ErrHTTPGetUserInfo + + +type alias UserInfo = + { name : String + , picture : String + } + + +type alias Configuration = + { authorizationEndpoint : Url + , tokenEndpoint : Url + , userInfoEndpoint : Url + , userInfoDecoder : Json.Decoder UserInfo + , clientId : String + , scope : List String + } + + +{-| During the authentication flow, we'll run twice into the `init` function: + + - The first time, for the application very first run. And we proceed with the `Idle` state, + waiting for the user (a.k.a you) to request a sign in. + + - The second time, after a sign in has been requested, the user is redirected to the + authorization server and redirects the user back to our application, with a code + and other fields as query parameters. + +When query params are present (and valid), we consider the user `Authorized`. + +-} +init : Maybe { state : String, codeVerifier : OAuth.CodeVerifier } -> Url -> Key -> ( Model, Cmd Msg ) +init mflags origin navigationKey = + let + redirectUri = + { origin | query = Nothing, fragment = Nothing } + + clearUrl = + Navigation.replaceUrl navigationKey (Url.toString redirectUri) + in + case OAuth.parseCode origin of + OAuth.Empty -> + ( { flow = Idle, redirectUri = redirectUri } + , Cmd.none + ) + + -- It is important to set a `state` when making the authorization request + -- and to verify it after the redirection. The state can be anything but its primary + -- usage is to prevent cross-site request forgery; at minima, it should be a short, + -- non-guessable string, generated on the fly. + -- + -- We remember any previously generated state state using the browser's local storage + -- and give it back (if present) to the elm application upon start + OAuth.Success { code, state } -> + case mflags of + Nothing -> + ( { flow = Errored ErrStateMismatch, redirectUri = redirectUri } + , clearUrl + ) + + Just flags -> + if state /= Just flags.state then + ( { flow = Errored ErrStateMismatch, redirectUri = redirectUri } + , clearUrl + ) + + else + ( { flow = Authorized code flags.codeVerifier, redirectUri = redirectUri } + , Cmd.batch + -- Artificial delay to make the live demo easier to follow. + -- In practice, the access token could be requested right here. + [ after 750 Millisecond AccessTokenRequested + , clearUrl + ] + ) + + OAuth.Error error -> + ( { flow = Errored <| ErrAuthorization error, redirectUri = redirectUri } + , clearUrl + ) + + + +-- +-- Msg +-- + + +type Msg + = NoOp + | SignInRequested + | GotRandomBytes (List Int) + | AccessTokenRequested + | GotAccessToken (Result Http.Error OAuth.AuthenticationSuccess) + | UserInfoRequested + | GotUserInfo (Result Http.Error UserInfo) + | SignOutRequested + + +getAccessToken : Configuration -> Url -> OAuth.AuthorizationCode -> OAuth.CodeVerifier -> Cmd Msg +getAccessToken { clientId, tokenEndpoint } redirectUri code codeVerifier = + Http.request <| + OAuth.makeTokenRequest GotAccessToken + { credentials = + { clientId = clientId + , secret = Nothing + } + , code = code + , codeVerifier = codeVerifier + , url = tokenEndpoint + , redirectUri = redirectUri + } + + +getUserInfo : Configuration -> OAuth.Token -> Cmd Msg +getUserInfo { userInfoDecoder, userInfoEndpoint } token = + Http.request + { method = "GET" + , body = Http.emptyBody + , headers = OAuth.useToken token [] + , url = Url.toString userInfoEndpoint + , expect = Http.expectJson GotUserInfo userInfoDecoder + , timeout = Nothing + , tracker = Nothing + } + + +{-| On the JavaScript's side, we have: + + app.ports.genRandomBytes.subscribe(n => { + const buffer = new Uint8Array(n); + crypto.getRandomValues(buffer); + const bytes = Array.from(buffer); + localStorage.setItem("bytes", bytes); + app.ports.randomBytes.send(bytes); + }); + +-} +port genRandomBytes : Int -> Cmd msg + + +port randomBytes : (List Int -> msg) -> Sub msg + + + +-- +-- Update +-- + + +update : Msg -> Model -> ( Model, Cmd Msg ) +update msg model = + case ( model.flow, msg ) of + ( Idle, SignInRequested ) -> + signInRequested model + + ( Idle, GotRandomBytes bytes ) -> + gotRandomBytes model bytes + + ( Authorized code codeVerifier, AccessTokenRequested ) -> + accessTokenRequested model code codeVerifier + + ( Authorized _ _, GotAccessToken authenticationResponse ) -> + gotAccessToken model authenticationResponse + + ( Authenticated token, UserInfoRequested ) -> + userInfoRequested model token + + ( Authenticated _, GotUserInfo userInfoResponse ) -> + gotUserInfo model userInfoResponse + + ( Done _, SignOutRequested ) -> + signOutRequested model + + _ -> + noOp model + + +noOp : Model -> ( Model, Cmd Msg ) +noOp model = + ( model, Cmd.none ) + + +signInRequested : Model -> ( Model, Cmd Msg ) +signInRequested model = + ( { model | flow = Idle } + -- We generate random bytes for both the state and the code verifier. First bytes are + -- for the 'state', and remaining ones are used for the code verifier. + , genRandomBytes (cSTATE_SIZE + cCODE_VERIFIER_SIZE) + ) + + +gotRandomBytes : Model -> List Int -> ( Model, Cmd Msg ) +gotRandomBytes model bytes = + case convertBytes bytes of + Nothing -> + ( { model | flow = Errored ErrFailedToConvertBytes } + , Cmd.none + ) + + Just { state, codeVerifier } -> + let + authorization = + { clientId = configuration.clientId + , redirectUri = model.redirectUri + , scope = configuration.scope + , state = Just state + , codeChallenge = OAuth.mkCodeChallenge codeVerifier + , url = configuration.authorizationEndpoint + } + in + ( { model | flow = Idle } + , authorization + |> OAuth.makeAuthorizationUrl + |> Url.toString + |> Navigation.load + ) + + +accessTokenRequested : Model -> OAuth.AuthorizationCode -> OAuth.CodeVerifier -> ( Model, Cmd Msg ) +accessTokenRequested model code codeVerifier = + ( { model | flow = Authorized code codeVerifier } + , getAccessToken configuration model.redirectUri code codeVerifier + ) + + +gotAccessToken : Model -> Result Http.Error OAuth.AuthenticationSuccess -> ( Model, Cmd Msg ) +gotAccessToken model authenticationResponse = + case authenticationResponse of + Err (Http.BadBody body) -> + case Json.decodeString OAuth.defaultAuthenticationErrorDecoder body of + Ok error -> + ( { model | flow = Errored <| ErrAuthentication error } + , Cmd.none + ) + + _ -> + ( { model | flow = Errored ErrHTTPGetAccessToken } + , Cmd.none + ) + + Err _ -> + ( { model | flow = Errored ErrHTTPGetAccessToken } + , Cmd.none + ) + + Ok { token } -> + ( { model | flow = Authenticated token } + , after 750 Millisecond UserInfoRequested + ) + + +userInfoRequested : Model -> OAuth.Token -> ( Model, Cmd Msg ) +userInfoRequested model token = + ( { model | flow = Authenticated token } + , getUserInfo configuration token + ) + + +gotUserInfo : Model -> Result Http.Error UserInfo -> ( Model, Cmd Msg ) +gotUserInfo model userInfoResponse = + case userInfoResponse of + Err _ -> + ( { model | flow = Errored ErrHTTPGetUserInfo } + , Cmd.none + ) + + Ok userInfo -> + ( { model | flow = Done userInfo } + , Cmd.none + ) + + +signOutRequested : Model -> ( Model, Cmd Msg ) +signOutRequested model = + ( { model | flow = Idle } + , Navigation.load (Url.toString model.redirectUri) + ) + + + +-- +-- View +-- + + +type alias ViewConfiguration msg = + { title : String + , btnClass : Attribute msg + } + + +view : ViewConfiguration Msg -> Model -> Document Msg +view ({ title } as config) model = + { title = title + , body = viewBody config model + } + + +viewBody : ViewConfiguration Msg -> Model -> List (Html Msg) +viewBody config model = + [ div [ class "flex", class "flex-column", class "flex-space-around" ] <| + case model.flow of + Idle -> + div [ class "flex" ] + [ viewAuthorizationStep False + , viewStepSeparator False + , viewAuthenticationStep False + , viewStepSeparator False + , viewGetUserInfoStep False + ] + :: viewIdle config + + Authorized _ _ -> + div [ class "flex" ] + [ viewAuthorizationStep True + , viewStepSeparator True + , viewAuthenticationStep False + , viewStepSeparator False + , viewGetUserInfoStep False + ] + :: viewAuthorized + + Authenticated _ -> + div [ class "flex" ] + [ viewAuthorizationStep True + , viewStepSeparator True + , viewAuthenticationStep True + , viewStepSeparator True + , viewGetUserInfoStep False + ] + :: viewAuthenticated + + Done userInfo -> + div [ class "flex" ] + [ viewAuthorizationStep True + , viewStepSeparator True + , viewAuthenticationStep True + , viewStepSeparator True + , viewGetUserInfoStep True + ] + :: viewUserInfo config userInfo + + Errored err -> + div [ class "flex" ] + [ viewErroredStep + ] + :: viewErrored err + ] + + +viewIdle : ViewConfiguration Msg -> List (Html Msg) +viewIdle { btnClass } = + [ button + [ onClick SignInRequested, btnClass ] + [ text "Sign in" ] + ] + + +viewAuthorized : List (Html Msg) +viewAuthorized = + [ span [] [ text "Authenticating..." ] + ] + + +viewAuthenticated : List (Html Msg) +viewAuthenticated = + [ span [] [ text "Getting user info..." ] + ] + + +viewUserInfo : ViewConfiguration Msg -> UserInfo -> List (Html Msg) +viewUserInfo { btnClass } { name, picture } = + [ div [ class "flex", class "flex-column" ] + [ img [ class "avatar", src picture ] [] + , p [] [ text name ] + , div [] + [ button + [ onClick SignOutRequested, btnClass ] + [ text "Sign out" ] + ] + ] + ] + + +viewErrored : Error -> List (Html Msg) +viewErrored error = + [ span [ class "span-error" ] [ viewError error ] ] + + +viewError : Error -> Html Msg +viewError e = + text <| + case e of + ErrStateMismatch -> + "'state' doesn't match, the request has likely been forged by an adversary!" + + ErrFailedToConvertBytes -> + "Unable to convert bytes to 'state' and 'codeVerifier', this is likely not your fault..." + + ErrAuthorization error -> + oauthErrorToString { error = error.error, errorDescription = error.errorDescription } + + ErrAuthentication error -> + oauthErrorToString { error = error.error, errorDescription = error.errorDescription } + + ErrHTTPGetAccessToken -> + "Unable to retrieve token: HTTP request failed. CORS is likely disabled on the authorization server." + + ErrHTTPGetUserInfo -> + "Unable to retrieve user info: HTTP request failed." + + +viewAuthorizationStep : Bool -> Html Msg +viewAuthorizationStep isActive = + viewStep isActive ( "Authorization", style "left" "-110%" ) + + +viewAuthenticationStep : Bool -> Html Msg +viewAuthenticationStep isActive = + viewStep isActive ( "Authentication", style "left" "-125%" ) + + +viewGetUserInfoStep : Bool -> Html Msg +viewGetUserInfoStep isActive = + viewStep isActive ( "Get User Info", style "left" "-135%" ) + + +viewErroredStep : Html Msg +viewErroredStep = + div + [ class "step", class "step-errored" ] + [ span [ style "left" "-50%" ] [ text "Errored" ] ] + + +viewStep : Bool -> ( String, Attribute Msg ) -> Html Msg +viewStep isActive ( step, position ) = + let + stepClass = + class "step" + :: (if isActive then + [ class "step-active" ] + + else + [] + ) + in + div stepClass [ span [ position ] [ text step ] ] + + +viewStepSeparator : Bool -> Html Msg +viewStepSeparator isActive = + let + stepClass = + class "step-separator" + :: (if isActive then + [ class "step-active" ] + + else + [] + ) + in + span stepClass [] + + + +-- +-- Helpers +-- +-- Number of bytes making the 'state' + + +cSTATE_SIZE : Int +cSTATE_SIZE = + 8 + + + +-- Number of bytes making the 'code_verifier' + + +cCODE_VERIFIER_SIZE : Int +cCODE_VERIFIER_SIZE = + 32 + + +toBytes : List Int -> Bytes +toBytes = + List.map Bytes.unsignedInt8 >> Bytes.sequence >> Bytes.encode + + +base64 : Bytes -> String +base64 = + Base64.bytes >> Base64.encode + + +convertBytes : List Int -> Maybe { state : String, codeVerifier : OAuth.CodeVerifier } +convertBytes bytes = + if List.length bytes < (cSTATE_SIZE + cCODE_VERIFIER_SIZE) then + Nothing + + else + let + state = + bytes + |> List.take cSTATE_SIZE + |> toBytes + |> base64 + + mCodeVerifier = + bytes + |> List.drop cSTATE_SIZE + |> toBytes + |> OAuth.codeVerifierFromBytes + in + Maybe.map (\codeVerifier -> { state = state, codeVerifier = codeVerifier }) mCodeVerifier + + +oauthErrorToString : { error : OAuth.ErrorCode, errorDescription : Maybe String } -> String +oauthErrorToString { error, errorDescription } = + let + desc = + errorDescription |> Maybe.withDefault "" |> String.replace "+" " " + in + OAuth.errorCodeToString error ++ ": " ++ desc + + +defaultHttpsUrl : Url +defaultHttpsUrl = + { protocol = Https + , host = "" + , path = "" + , port_ = Nothing + , query = Nothing + , fragment = Nothing + } diff --git a/examples/providers/auth0/pkce/README.md b/examples/providers/auth0/pkce/README.md new file mode 100644 index 0000000..e2e07a7 --- /dev/null +++ b/examples/providers/auth0/pkce/README.md @@ -0,0 +1,5 @@ +# Auth0 + +

+ +

diff --git a/examples/providers/auth0/pkce/elm.json b/examples/providers/auth0/pkce/elm.json new file mode 100644 index 0000000..a933f27 --- /dev/null +++ b/examples/providers/auth0/pkce/elm.json @@ -0,0 +1,34 @@ +{ + "type": "application", + "source-directories": [ + "src", + "." + ], + "elm-version": "0.19.1", + "dependencies": { + "direct": { + "andrewMacmurray/elm-delay": "3.0.0", + "elm/browser": "1.0.2", + "elm/bytes": "1.0.8", + "elm/core": "1.0.4", + "elm/html": "1.0.0", + "elm/http": "2.0.0", + "elm/json": "1.1.3", + "elm/url": "1.0.0", + "folkertdev/elm-sha2": "1.0.0", + "ivadzy/bbase64": "1.1.1" + }, + "indirect": { + "danfishgold/base64-bytes": "1.0.3", + "elm/file": "1.0.5", + "elm/regex": "1.0.0", + "elm/time": "1.0.0", + "elm/virtual-dom": "1.0.2", + "rtfeldman/elm-hex": "1.0.0" + } + }, + "test-dependencies": { + "direct": {}, + "indirect": {} + } +} diff --git a/examples/providers/auth0/pkce/src b/examples/providers/auth0/pkce/src new file mode 120000 index 0000000..dd81945 --- /dev/null +++ b/examples/providers/auth0/pkce/src @@ -0,0 +1 @@ +../../../../src/ \ No newline at end of file diff --git a/examples/providers/facebook/README.md b/examples/providers/facebook/README.md new file mode 100644 index 0000000..c7e91f6 --- /dev/null +++ b/examples/providers/facebook/README.md @@ -0,0 +1,27 @@ +# Facebook + +## Authorization Flows + +Flow | Support | Remark | Example +--- | --- | --- | --- +Implicit | :heavy_check_mark: | Non-standard parsers required | [live demo][implicit-demo] \| [source code][implicit-source] +Authorization Code | :heavy_check_mark: | Non-standard parsers required
Requires secret key (server-side) | N/A +Authorization Code w/ PKCE | :x: | \- | \- +Password | :heavy_check_mark: | \- | \- +Client Credentials | :heavy_check_mark: | Requires secret key (server-side) | N/A + +## OAuth Configuration + +\- | \- +--- | --- +Authorization Endpoint | facebook.com/v6.0/dialog/oauth +Token Endpoint | graph.facebook.com/v6.0/oauth/access_token +User Info Endpoint | graph.facebook.com/v6.0/me + +--- + +:book: https://developers.facebook.com/docs/facebook-login/manually-build-a-login-flow + + +[implicit-demo]: https://truqu.github.io/elm-oauth2/facebook/implicit/ +[implicit-source]: https://github.com/truqu/elm-oauth2/blob/master/examples/providers/facebook/implicit/Main.elm diff --git a/examples/providers/facebook/implicit/Main.elm b/examples/providers/facebook/implicit/Main.elm new file mode 100644 index 0000000..647fb8a --- /dev/null +++ b/examples/providers/facebook/implicit/Main.elm @@ -0,0 +1,558 @@ +port module Main exposing (main) + +import Base64.Encode as Base64 +import Browser exposing (Document, application) +import Browser.Navigation as Navigation exposing (Key) +import Bytes exposing (Bytes) +import Bytes.Encode as Bytes +import Delay exposing (TimeUnit(..), after) +import Html exposing (..) +import Html.Attributes exposing (..) +import Html.Events exposing (..) +import Http +import Json.Decode as Json +import OAuth +import OAuth.Implicit as OAuth +import Url exposing (Protocol(..), Url) +import Url.Parser.Query as Query + + +main : Program (Maybe (List Int)) Model Msg +main = + application + { init = + Maybe.map convertBytes >> init + , update = + update + , subscriptions = + always <| randomBytes GotRandomBytes + , onUrlRequest = + always NoOp + , onUrlChange = + always NoOp + , view = + view + { title = "Facebook - Flow: Implicit" + , btnClass = class "btn-facebook" + } + } + + +{-| OAuth configuration. + +Note that this demo also fetches basic user information with the obtained access token, +hence the user info endpoint and JSON decoder + +-} +configuration : Configuration +configuration = + { authorizationEndpoint = + { defaultHttpsUrl | host = "facebook.com", path = "/v6.0/dialog/oauth" } + , userInfoEndpoint = + { defaultHttpsUrl | host = "graph.facebook.com", path = "/v6.0/me", query = Just "fields=name,picture.type(large)" } + , userInfoDecoder = + Json.map2 UserInfo + (Json.field "name" Json.string) + (Json.field "picture" <| Json.field "data" <| Json.field "url" Json.string) + , clientId = + "179456896198275" + , scope = + [ "public_profile" ] + } + + + +-- +-- Model +-- + + +type alias Model = + { redirectUri : Url + , flow : Flow + } + + +{-| This demo evolves around the following state-machine\* + + +--------+ + | Idle | + +--------+ + | + | Redirect user for authorization + | + v + +--------------+ + | Authorized | w/ Access Token + +--------------+ + | + | Fetch user info using the access token + v + +--------+ + | Done | + +--------+ + +(\*) The 'Errored' state hasn't been represented here for simplicity. + +-} +type Flow + = Idle + | Authorized OAuth.Token + | Done UserInfo + | Errored Error + + +type Error + = ErrStateMismatch + | ErrAuthorization OAuth.AuthorizationError + | ErrHTTPGetUserInfo + + +type alias UserInfo = + { name : String + , picture : String + } + + +type alias Configuration = + { authorizationEndpoint : Url + , userInfoEndpoint : Url + , userInfoDecoder : Json.Decoder UserInfo + , clientId : String + , scope : List String + } + + +{-| During the authentication flow, we'll run twice into the `init` function: + + - The first time, for the application very first run. And we proceed with the `Idle` state, + waiting for the user (a.k.a you) to request a sign in. + + - The second time, after a sign in has been requested, the user is redirected to the + authorization server and redirects the user back to our application, with an access + token and other fields as query parameters. + +When query params are present (and valid), we consider the user `Authorized`. + +-} +init : Maybe { state : String } -> Url -> Key -> ( Model, Cmd Msg ) +init mflags origin navigationKey = + let + redirectUri = + { origin | query = Nothing, fragment = Nothing } + + clearUrl = + Navigation.replaceUrl navigationKey (Url.toString redirectUri) + in + case OAuth.parseTokenWith parsers (patchUrl origin) of + OAuth.Empty -> + ( { flow = Idle, redirectUri = redirectUri } + , Cmd.none + ) + + -- It is important to set a `state` when making the authorization request + -- and to verify it after the redirection. The state can be anything but its primary + -- usage is to prevent cross-site request forgery; at minima, it should be a short, + -- non-guessable string, generated on the fly. + -- + -- We remember any previously generated state state using the browser's local storage + -- and give it back (if present) to the elm application upon start + OAuth.Success { token, state } -> + case mflags of + Nothing -> + ( { flow = Errored ErrStateMismatch, redirectUri = redirectUri } + , clearUrl + ) + + Just flags -> + if state /= Just flags.state then + ( { flow = Errored ErrStateMismatch, redirectUri = redirectUri } + , clearUrl + ) + + else + ( { flow = Authorized token, redirectUri = redirectUri } + , Cmd.batch + -- Artificial delay to make the live demo easier to follow. + -- In practice, the access token could be requested right here. + [ after 750 Millisecond UserInfoRequested + , clearUrl + ] + ) + + OAuth.Error error -> + ( { flow = Errored <| ErrAuthorization error, redirectUri = redirectUri } + , clearUrl + ) + + + +-- +-- Msg +-- + + +type Msg + = NoOp + | SignInRequested + | GotRandomBytes (List Int) + | GotAccessToken (Result Http.Error OAuth.AuthorizationSuccess) + | UserInfoRequested + | GotUserInfo (Result Http.Error UserInfo) + | SignOutRequested + + +getUserInfo : Configuration -> OAuth.Token -> Cmd Msg +getUserInfo { userInfoDecoder, userInfoEndpoint } token = + Http.request + { method = "GET" + , body = Http.emptyBody + , headers = OAuth.useToken token [] + , url = Url.toString userInfoEndpoint + , expect = Http.expectJson GotUserInfo userInfoDecoder + , timeout = Nothing + , tracker = Nothing + } + + + +{- On the JavaScript's side, we have: + + app.ports.genRandomBytes.subscribe(n => { + const buffer = new Uint8Array(n); + crypto.getRandomValues(buffer); + const bytes = Array.from(buffer); + localStorage.setItem("bytes", bytes); + app.ports.randomBytes.send(bytes); + }); +-} + + +port genRandomBytes : Int -> Cmd msg + + +port randomBytes : (List Int -> msg) -> Sub msg + + + +-- +-- Update +-- + + +update : Msg -> Model -> ( Model, Cmd Msg ) +update msg model = + case ( model.flow, msg ) of + ( Idle, SignInRequested ) -> + signInRequested model + + ( Idle, GotRandomBytes bytes ) -> + gotRandomBytes model bytes + + ( Authorized token, UserInfoRequested ) -> + userInfoRequested model token + + ( Authorized _, GotUserInfo userInfoResponse ) -> + gotUserInfo model userInfoResponse + + ( Done _, SignOutRequested ) -> + signOutRequested model + + _ -> + noOp model + + +noOp : Model -> ( Model, Cmd Msg ) +noOp model = + ( model, Cmd.none ) + + +signInRequested : Model -> ( Model, Cmd Msg ) +signInRequested model = + ( { model | flow = Idle } + , genRandomBytes 16 + ) + + +gotRandomBytes : Model -> List Int -> ( Model, Cmd Msg ) +gotRandomBytes model bytes = + let + { state } = + convertBytes bytes + + authorization = + { clientId = configuration.clientId + , redirectUri = model.redirectUri + , scope = configuration.scope + , state = Just state + , url = configuration.authorizationEndpoint + } + in + ( { model | flow = Idle } + , authorization + |> OAuth.makeAuthorizationUrl + |> Url.toString + |> Navigation.load + ) + + +userInfoRequested : Model -> OAuth.Token -> ( Model, Cmd Msg ) +userInfoRequested model token = + ( { model | flow = Authorized token } + , getUserInfo configuration token + ) + + +gotUserInfo : Model -> Result Http.Error UserInfo -> ( Model, Cmd Msg ) +gotUserInfo model userInfoResponse = + case userInfoResponse of + Err _ -> + ( { model | flow = Errored ErrHTTPGetUserInfo } + , Cmd.none + ) + + Ok userInfo -> + ( { model | flow = Done userInfo } + , Cmd.none + ) + + +signOutRequested : Model -> ( Model, Cmd Msg ) +signOutRequested model = + ( { model | flow = Idle } + , Navigation.load (Url.toString model.redirectUri) + ) + + + +-- +-- View +-- + + +type alias ViewConfiguration msg = + { title : String + , btnClass : Attribute msg + } + + +view : ViewConfiguration Msg -> Model -> Document Msg +view ({ title } as config) model = + { title = title + , body = viewBody config model + } + + +viewBody : ViewConfiguration Msg -> Model -> List (Html Msg) +viewBody config model = + [ div [ class "flex", class "flex-column", class "flex-space-around" ] <| + case model.flow of + Idle -> + div [ class "flex" ] + [ viewAuthorizationStep False + , viewStepSeparator False + , viewGetUserInfoStep False + ] + :: viewIdle config + + Authorized _ -> + div [ class "flex" ] + [ viewAuthorizationStep True + , viewStepSeparator True + , viewGetUserInfoStep False + ] + :: viewAuthorized + + Done userInfo -> + div [ class "flex" ] + [ viewAuthorizationStep True + , viewStepSeparator True + , viewGetUserInfoStep True + ] + :: viewUserInfo config userInfo + + Errored err -> + div [ class "flex" ] + [ viewErroredStep + ] + :: viewErrored err + ] + + +viewIdle : ViewConfiguration Msg -> List (Html Msg) +viewIdle { btnClass } = + [ button + [ onClick SignInRequested, btnClass ] + [ text "Sign in" ] + ] + + +viewAuthorized : List (Html Msg) +viewAuthorized = + [ span [] [ text "Getting user info..." ] + ] + + +viewUserInfo : ViewConfiguration Msg -> UserInfo -> List (Html Msg) +viewUserInfo { btnClass } { name, picture } = + [ div [ class "flex", class "flex-column" ] + [ img [ class "avatar", src picture ] [] + , p [] [ text name ] + , div [] + [ button + [ onClick SignOutRequested, btnClass ] + [ text "Sign out" ] + ] + ] + ] + + +viewErrored : Error -> List (Html Msg) +viewErrored error = + [ span [ class "span-error" ] [ viewError error ] ] + + +viewError : Error -> Html Msg +viewError e = + text <| + case e of + ErrStateMismatch -> + "'state' doesn't match, the request has likely been forged by an adversary!" + + ErrAuthorization error -> + oauthErrorToString { error = error.error, errorDescription = error.errorDescription } + + ErrHTTPGetUserInfo -> + "Unable to retrieve user info: HTTP request failed." + + +viewAuthorizationStep : Bool -> Html Msg +viewAuthorizationStep isActive = + viewStep isActive ( "Authorization", style "left" "-110%" ) + + +viewGetUserInfoStep : Bool -> Html Msg +viewGetUserInfoStep isActive = + viewStep isActive ( "Get User Info", style "left" "-135%" ) + + +viewErroredStep : Html Msg +viewErroredStep = + div + [ class "step", class "step-errored" ] + [ span [ style "left" "-50%" ] [ text "Errored" ] ] + + +viewStep : Bool -> ( String, Attribute Msg ) -> Html Msg +viewStep isActive ( step, position ) = + let + stepClass = + class "step" + :: (if isActive then + [ class "step-active" ] + + else + [] + ) + in + div stepClass [ span [ position ] [ text step ] ] + + +viewStepSeparator : Bool -> Html Msg +viewStepSeparator isActive = + let + stepClass = + class "step-separator" + :: (if isActive then + [ class "step-active" ] + + else + [] + ) + in + span stepClass [] + + + +-- +-- Facebook Wrong Implementation Work-Arounds +-- + + +{-| No 'token\_type' is returned, so we have to provide a default one as Just "Bearer". +-} +tokenParser : Query.Parser (Maybe OAuth.Token) +tokenParser = + Query.map (OAuth.makeToken (Just "Bearer")) + (Query.string "access_token") + + +{-| In case of error, no 'error' field is returned, but instead we find a field named 'error\_code' +-} +errorParser : Query.Parser (Maybe OAuth.ErrorCode) +errorParser = + Query.map (Maybe.map OAuth.errorCodeFromString) + (Query.string "error_code") + + +{-| Put everything together and rely on `OAuth.parseTokenWith` instead of the default parser +-} +parsers : OAuth.Parsers +parsers = + { tokenParser = tokenParser + , errorParser = errorParser + , authorizationSuccessParser = OAuth.defaultAuthorizationSuccessParser + , authorizationErrorParser = OAuth.defaultAuthorizationErrorParser + } + + +{-| In addition, Facebook returns parameters as query parameters instead of a fragments, and sometimes, a noise fragment is present in the response. So, as a work-around, one can patch the Url to make it compliant with the original RFC specification as follows: +-} +patchUrl : Url -> Url +patchUrl url = + if url.fragment == Just "_=_" || url.fragment == Nothing then + { url | fragment = url.query } + + else + url + + + +-- +-- Helpers +-- + + +toBytes : List Int -> Bytes +toBytes = + List.map Bytes.unsignedInt8 >> Bytes.sequence >> Bytes.encode + + +base64 : Bytes -> String +base64 = + Base64.bytes >> Base64.encode + + +convertBytes : List Int -> { state : String } +convertBytes = + toBytes >> base64 >> (\state -> { state = state }) + + +oauthErrorToString : { error : OAuth.ErrorCode, errorDescription : Maybe String } -> String +oauthErrorToString { error, errorDescription } = + let + desc = + errorDescription |> Maybe.withDefault "" |> String.replace "+" " " + in + OAuth.errorCodeToString error ++ ": " ++ desc + + +defaultHttpsUrl : Url +defaultHttpsUrl = + { protocol = Https + , host = "" + , path = "" + , port_ = Nothing + , query = Nothing + , fragment = Nothing + } diff --git a/examples/providers/facebook/implicit/elm.json b/examples/providers/facebook/implicit/elm.json new file mode 100644 index 0000000..a933f27 --- /dev/null +++ b/examples/providers/facebook/implicit/elm.json @@ -0,0 +1,34 @@ +{ + "type": "application", + "source-directories": [ + "src", + "." + ], + "elm-version": "0.19.1", + "dependencies": { + "direct": { + "andrewMacmurray/elm-delay": "3.0.0", + "elm/browser": "1.0.2", + "elm/bytes": "1.0.8", + "elm/core": "1.0.4", + "elm/html": "1.0.0", + "elm/http": "2.0.0", + "elm/json": "1.1.3", + "elm/url": "1.0.0", + "folkertdev/elm-sha2": "1.0.0", + "ivadzy/bbase64": "1.1.1" + }, + "indirect": { + "danfishgold/base64-bytes": "1.0.3", + "elm/file": "1.0.5", + "elm/regex": "1.0.0", + "elm/time": "1.0.0", + "elm/virtual-dom": "1.0.2", + "rtfeldman/elm-hex": "1.0.0" + } + }, + "test-dependencies": { + "direct": {}, + "indirect": {} + } +} diff --git a/examples/providers/facebook/implicit/src b/examples/providers/facebook/implicit/src new file mode 120000 index 0000000..dd81945 --- /dev/null +++ b/examples/providers/facebook/implicit/src @@ -0,0 +1 @@ +../../../../src/ \ No newline at end of file diff --git a/examples/providers/google/README.md b/examples/providers/google/README.md new file mode 100644 index 0000000..2d0f312 --- /dev/null +++ b/examples/providers/google/README.md @@ -0,0 +1,27 @@ +# Google + +## Authorization Flows + +Flow | Support | Remark | Example +--- | --- | --- | --- +Implicit | :heavy_check_mark: | \- | [live demo][implicit-demo] \| [source code][implicit-source] +Authorization Code | :heavy_check_mark: | Requires secret key (server-side) | N/A +Authorization Code w/ PKCE | :heavy_check_mark: | Requires secret key (server-side) | N/A +Password | :heavy_check_mark: | Requires secret key (server-side) | N/A +Client Credentials | :heavy_check_mark: | Requires secret key (server-side) | N/A + +## OAuth Configuration + +\- | \- +--- | --- +Authorization Endpoint | accounts.google.com/o/oauth2/v2/auth +Token Endpoint | www.googleapis.com/oauth2/v4/token +User Info Endpoint | www.googleapis.com/oauth2/v1/userinfo + +--- + +:book: https://developers.google.com/identity/protocols/OAuth2 + + +[implicit-demo]: https://truqu.github.io/elm-oauth2/google/implicit/ +[implicit-source]: https://github.com/truqu/elm-oauth2/blob/master/examples/providers/google/implicit/Main.elm diff --git a/examples/providers/google/implicit/Main.elm b/examples/providers/google/implicit/Main.elm new file mode 100644 index 0000000..74e0df1 --- /dev/null +++ b/examples/providers/google/implicit/Main.elm @@ -0,0 +1,513 @@ +port module Main exposing (main) + +import Base64.Encode as Base64 +import Browser exposing (Document, application) +import Browser.Navigation as Navigation exposing (Key) +import Bytes exposing (Bytes) +import Bytes.Encode as Bytes +import Delay exposing (TimeUnit(..), after) +import Html exposing (..) +import Html.Attributes exposing (..) +import Html.Events exposing (..) +import Http +import Json.Decode as Json +import OAuth +import OAuth.Implicit as OAuth +import Url exposing (Protocol(..), Url) + + +main : Program (Maybe (List Int)) Model Msg +main = + application + { init = + Maybe.map convertBytes >> init + , update = + update + , subscriptions = + always <| randomBytes GotRandomBytes + , onUrlRequest = + always NoOp + , onUrlChange = + always NoOp + , view = + view + { title = "Google - Flow: Implicit" + , btnClass = class "btn-google" + } + } + + +{-| OAuth configuration. + +Note that this demo also fetches basic user information with the obtained access token, +hence the user info endpoint and JSON decoder + +-} +configuration : Configuration +configuration = + { authorizationEndpoint = + { defaultHttpsUrl | host = "accounts.google.com", path = "/o/oauth2/v2/auth" } + , userInfoEndpoint = + { defaultHttpsUrl | host = "www.googleapis.com", path = "/oauth2/v1/userinfo" } + , userInfoDecoder = + Json.map2 UserInfo + (Json.field "name" Json.string) + (Json.field "picture" Json.string) + , clientId = + "909608474358-fkok86ks7e83c47aq01aiit47vsoh4s0.apps.googleusercontent.com" + , scope = + [ "profile" ] + } + + + +-- +-- Model +-- + + +type alias Model = + { redirectUri : Url + , flow : Flow + } + + +{-| This demo evolves around the following state-machine\* + + +--------+ + | Idle | + +--------+ + | + | Redirect user for authorization + | + v + +--------------+ + | Authorized | w/ Access Token + +--------------+ + | + | Fetch user info using the access token + v + +--------+ + | Done | + +--------+ + +(\*) The 'Errored' state hasn't been represented here for simplicity. + +-} +type Flow + = Idle + | Authorized OAuth.Token + | Done UserInfo + | Errored Error + + +type Error + = ErrStateMismatch + | ErrAuthorization OAuth.AuthorizationError + | ErrHTTPGetUserInfo + + +type alias UserInfo = + { name : String + , picture : String + } + + +type alias Configuration = + { authorizationEndpoint : Url + , userInfoEndpoint : Url + , userInfoDecoder : Json.Decoder UserInfo + , clientId : String + , scope : List String + } + + +{-| During the authentication flow, we'll run twice into the `init` function: + + - The first time, for the application very first run. And we proceed with the `Idle` state, + waiting for the user (a.k.a you) to request a sign in. + + - The second time, after a sign in has been requested, the user is redirected to the + authorization server and redirects the user back to our application, with an access + token and other fields as query parameters. + +When query params are present (and valid), we consider the user `Authorized`. + +-} +init : Maybe { state : String } -> Url -> Key -> ( Model, Cmd Msg ) +init mflags origin navigationKey = + let + redirectUri = + { origin | query = Nothing, fragment = Nothing } + + clearUrl = + Navigation.replaceUrl navigationKey (Url.toString redirectUri) + in + case OAuth.parseToken origin of + OAuth.Empty -> + ( { flow = Idle, redirectUri = redirectUri } + , Cmd.none + ) + + -- It is important to set a `state` when making the authorization request + -- and to verify it after the redirection. The state can be anything but its primary + -- usage is to prevent cross-site request forgery; at minima, it should be a short, + -- non-guessable string, generated on the fly. + -- + -- We remember any previously generated state state using the browser's local storage + -- and give it back (if present) to the elm application upon start + OAuth.Success { token, state } -> + case mflags of + Nothing -> + ( { flow = Errored ErrStateMismatch, redirectUri = redirectUri } + , clearUrl + ) + + Just flags -> + if state /= Just flags.state then + ( { flow = Errored ErrStateMismatch, redirectUri = redirectUri } + , clearUrl + ) + + else + ( { flow = Authorized token, redirectUri = redirectUri } + , Cmd.batch + -- Artificial delay to make the live demo easier to follow. + -- In practice, the access token could be requested right here. + [ after 750 Millisecond UserInfoRequested + , clearUrl + ] + ) + + OAuth.Error error -> + ( { flow = Errored <| ErrAuthorization error, redirectUri = redirectUri } + , clearUrl + ) + + + +-- +-- Msg +-- + + +type Msg + = NoOp + | SignInRequested + | GotRandomBytes (List Int) + | GotAccessToken (Result Http.Error OAuth.AuthorizationSuccess) + | UserInfoRequested + | GotUserInfo (Result Http.Error UserInfo) + | SignOutRequested + + +getUserInfo : Configuration -> OAuth.Token -> Cmd Msg +getUserInfo { userInfoDecoder, userInfoEndpoint } token = + Http.request + { method = "GET" + , body = Http.emptyBody + , headers = OAuth.useToken token [] + , url = Url.toString userInfoEndpoint + , expect = Http.expectJson GotUserInfo userInfoDecoder + , timeout = Nothing + , tracker = Nothing + } + + + +{- On the JavaScript's side, we have: + + app.ports.genRandomBytes.subscribe(n => { + const buffer = new Uint8Array(n); + crypto.getRandomValues(buffer); + const bytes = Array.from(buffer); + localStorage.setItem("bytes", bytes); + app.ports.randomBytes.send(bytes); + }); +-} + + +port genRandomBytes : Int -> Cmd msg + + +port randomBytes : (List Int -> msg) -> Sub msg + + + +-- +-- Update +-- + + +update : Msg -> Model -> ( Model, Cmd Msg ) +update msg model = + case ( model.flow, msg ) of + ( Idle, SignInRequested ) -> + signInRequested model + + ( Idle, GotRandomBytes bytes ) -> + gotRandomBytes model bytes + + ( Authorized token, UserInfoRequested ) -> + userInfoRequested model token + + ( Authorized _, GotUserInfo userInfoResponse ) -> + gotUserInfo model userInfoResponse + + ( Done _, SignOutRequested ) -> + signOutRequested model + + _ -> + noOp model + + +noOp : Model -> ( Model, Cmd Msg ) +noOp model = + ( model, Cmd.none ) + + +signInRequested : Model -> ( Model, Cmd Msg ) +signInRequested model = + ( { model | flow = Idle } + , genRandomBytes 16 + ) + + +gotRandomBytes : Model -> List Int -> ( Model, Cmd Msg ) +gotRandomBytes model bytes = + let + { state } = + convertBytes bytes + + authorization = + { clientId = configuration.clientId + , redirectUri = model.redirectUri + , scope = configuration.scope + , state = Just state + , url = configuration.authorizationEndpoint + } + in + ( { model | flow = Idle } + , authorization + |> OAuth.makeAuthorizationUrl + |> Url.toString + |> Navigation.load + ) + + +userInfoRequested : Model -> OAuth.Token -> ( Model, Cmd Msg ) +userInfoRequested model token = + ( { model | flow = Authorized token } + , getUserInfo configuration token + ) + + +gotUserInfo : Model -> Result Http.Error UserInfo -> ( Model, Cmd Msg ) +gotUserInfo model userInfoResponse = + case userInfoResponse of + Err _ -> + ( { model | flow = Errored ErrHTTPGetUserInfo } + , Cmd.none + ) + + Ok userInfo -> + ( { model | flow = Done userInfo } + , Cmd.none + ) + + +signOutRequested : Model -> ( Model, Cmd Msg ) +signOutRequested model = + ( { model | flow = Idle } + , Navigation.load (Url.toString model.redirectUri) + ) + + + +-- +-- View +-- + + +type alias ViewConfiguration msg = + { title : String + , btnClass : Attribute msg + } + + +view : ViewConfiguration Msg -> Model -> Document Msg +view ({ title } as config) model = + { title = title + , body = viewBody config model + } + + +viewBody : ViewConfiguration Msg -> Model -> List (Html Msg) +viewBody config model = + [ div [ class "flex", class "flex-column", class "flex-space-around" ] <| + case model.flow of + Idle -> + div [ class "flex" ] + [ viewAuthorizationStep False + , viewStepSeparator False + , viewGetUserInfoStep False + ] + :: viewIdle config + + Authorized _ -> + div [ class "flex" ] + [ viewAuthorizationStep True + , viewStepSeparator True + , viewGetUserInfoStep False + ] + :: viewAuthorized + + Done userInfo -> + div [ class "flex" ] + [ viewAuthorizationStep True + , viewStepSeparator True + , viewGetUserInfoStep True + ] + :: viewUserInfo config userInfo + + Errored err -> + div [ class "flex" ] + [ viewErroredStep + ] + :: viewErrored err + ] + + +viewIdle : ViewConfiguration Msg -> List (Html Msg) +viewIdle { btnClass } = + [ button + [ onClick SignInRequested, btnClass ] + [ text "Sign in" ] + ] + + +viewAuthorized : List (Html Msg) +viewAuthorized = + [ span [] [ text "Getting user info..." ] + ] + + +viewUserInfo : ViewConfiguration Msg -> UserInfo -> List (Html Msg) +viewUserInfo { btnClass } { name, picture } = + [ div [ class "flex", class "flex-column" ] + [ img [ class "avatar", src picture ] [] + , p [] [ text name ] + , div [] + [ button + [ onClick SignOutRequested, btnClass ] + [ text "Sign out" ] + ] + ] + ] + + +viewErrored : Error -> List (Html Msg) +viewErrored error = + [ span [ class "span-error" ] [ viewError error ] ] + + +viewError : Error -> Html Msg +viewError e = + text <| + case e of + ErrStateMismatch -> + "'state' doesn't match, the request has likely been forged by an adversary!" + + ErrAuthorization error -> + oauthErrorToString { error = error.error, errorDescription = error.errorDescription } + + ErrHTTPGetUserInfo -> + "Unable to retrieve user info: HTTP request failed." + + +viewAuthorizationStep : Bool -> Html Msg +viewAuthorizationStep isActive = + viewStep isActive ( "Authorization", style "left" "-110%" ) + + +viewGetUserInfoStep : Bool -> Html Msg +viewGetUserInfoStep isActive = + viewStep isActive ( "Get User Info", style "left" "-135%" ) + + +viewErroredStep : Html Msg +viewErroredStep = + div + [ class "step", class "step-errored" ] + [ span [ style "left" "-50%" ] [ text "Errored" ] ] + + +viewStep : Bool -> ( String, Attribute Msg ) -> Html Msg +viewStep isActive ( step, position ) = + let + stepClass = + class "step" + :: (if isActive then + [ class "step-active" ] + + else + [] + ) + in + div stepClass [ span [ position ] [ text step ] ] + + +viewStepSeparator : Bool -> Html Msg +viewStepSeparator isActive = + let + stepClass = + class "step-separator" + :: (if isActive then + [ class "step-active" ] + + else + [] + ) + in + span stepClass [] + + + +-- +-- Helpers +-- + + +toBytes : List Int -> Bytes +toBytes = + List.map Bytes.unsignedInt8 >> Bytes.sequence >> Bytes.encode + + +base64 : Bytes -> String +base64 = + Base64.bytes >> Base64.encode + + +convertBytes : List Int -> { state : String } +convertBytes = + toBytes >> base64 >> (\state -> { state = state }) + + +oauthErrorToString : { error : OAuth.ErrorCode, errorDescription : Maybe String } -> String +oauthErrorToString { error, errorDescription } = + let + desc = + errorDescription |> Maybe.withDefault "" |> String.replace "+" " " + in + OAuth.errorCodeToString error ++ ": " ++ desc + + +defaultHttpsUrl : Url +defaultHttpsUrl = + { protocol = Https + , host = "" + , path = "" + , port_ = Nothing + , query = Nothing + , fragment = Nothing + } diff --git a/examples/providers/google/implicit/README.md b/examples/providers/google/implicit/README.md new file mode 100644 index 0000000..e2e07a7 --- /dev/null +++ b/examples/providers/google/implicit/README.md @@ -0,0 +1,5 @@ +# Auth0 + +

+ +

diff --git a/examples/providers/google/implicit/elm.json b/examples/providers/google/implicit/elm.json new file mode 100644 index 0000000..a933f27 --- /dev/null +++ b/examples/providers/google/implicit/elm.json @@ -0,0 +1,34 @@ +{ + "type": "application", + "source-directories": [ + "src", + "." + ], + "elm-version": "0.19.1", + "dependencies": { + "direct": { + "andrewMacmurray/elm-delay": "3.0.0", + "elm/browser": "1.0.2", + "elm/bytes": "1.0.8", + "elm/core": "1.0.4", + "elm/html": "1.0.0", + "elm/http": "2.0.0", + "elm/json": "1.1.3", + "elm/url": "1.0.0", + "folkertdev/elm-sha2": "1.0.0", + "ivadzy/bbase64": "1.1.1" + }, + "indirect": { + "danfishgold/base64-bytes": "1.0.3", + "elm/file": "1.0.5", + "elm/regex": "1.0.0", + "elm/time": "1.0.0", + "elm/virtual-dom": "1.0.2", + "rtfeldman/elm-hex": "1.0.0" + } + }, + "test-dependencies": { + "direct": {}, + "indirect": {} + } +} diff --git a/examples/providers/google/implicit/src b/examples/providers/google/implicit/src new file mode 120000 index 0000000..dd81945 --- /dev/null +++ b/examples/providers/google/implicit/src @@ -0,0 +1 @@ +../../../../src/ \ No newline at end of file diff --git a/examples/providers/spotify/README.md b/examples/providers/spotify/README.md new file mode 100644 index 0000000..1256592 --- /dev/null +++ b/examples/providers/spotify/README.md @@ -0,0 +1,27 @@ +# Spotify + +## Authorization Flows + +Flow | Support | Remark | Example +--- | --- | --- | --- +Implicit | :heavy_check_mark: | \- | [live demo][implicit-demo] \| [source code][implicit-source] +Authorization Code | :heavy_check_mark: | Requires secret key (server-side) | N/A +Authorization Code w/ PKCE | :x: | \- | \- +Password | :x: | \- | \- +Client Credentials | :heavy_check_mark: | Requires secret key (server-side)
No access to user resources | N/A + +## OAuth Configuration + +\- | \- +--- | --- +Authorization Endpoint | accounts.spotify.com/authorize +Token Endpoint | accounts.spotify.com/api/token +User Info Endpoint | api.spotify.com/v1/me + +--- + +:book: https://developer.spotify.com/documentation/general/guides/authorization-guide/ + + +[implicit-demo]: https://truqu.github.io/elm-oauth2/spotify/implicit/ +[implicit-source]: https://github.com/truqu/elm-oauth2/blob/master/examples/providers/spotify/implicit/Main.elm diff --git a/examples/providers/spotify/implicit/Main.elm b/examples/providers/spotify/implicit/Main.elm new file mode 100644 index 0000000..8342f25 --- /dev/null +++ b/examples/providers/spotify/implicit/Main.elm @@ -0,0 +1,513 @@ +port module Main exposing (main) + +import Base64.Encode as Base64 +import Browser exposing (Document, application) +import Browser.Navigation as Navigation exposing (Key) +import Bytes exposing (Bytes) +import Bytes.Encode as Bytes +import Delay exposing (TimeUnit(..), after) +import Html exposing (..) +import Html.Attributes exposing (..) +import Html.Events exposing (..) +import Http +import Json.Decode as Json +import OAuth +import OAuth.Implicit as OAuth +import Url exposing (Protocol(..), Url) + + +main : Program (Maybe (List Int)) Model Msg +main = + application + { init = + Maybe.map convertBytes >> init + , update = + update + , subscriptions = + always <| randomBytes GotRandomBytes + , onUrlRequest = + always NoOp + , onUrlChange = + always NoOp + , view = + view + { title = "Spotify - Flow: Implicit" + , btnClass = class "btn-spotify" + } + } + + +{-| OAuth configuration. + +Note that this demo also fetches basic user information with the obtained access token, +hence the user info endpoint and JSON decoder + +-} +configuration : Configuration +configuration = + { authorizationEndpoint = + { defaultHttpsUrl | host = "accounts.spotify.com", path = "/authorize" } + , userInfoEndpoint = + { defaultHttpsUrl | host = "api.spotify.com", path = "/v1/me" } + , userInfoDecoder = + Json.map2 UserInfo + (Json.field "display_name" Json.string) + (Json.field "images" <| Json.index 0 <| Json.field "url" Json.string) + , clientId = + "391d08ef3d7a46558493cb822a991dbb" + , scope = + [] + } + + + +-- +-- Model +-- + + +type alias Model = + { redirectUri : Url + , flow : Flow + } + + +{-| This demo evolves around the following state-machine\* + + +--------+ + | Idle | + +--------+ + | + | Redirect user for authorization + | + v + +--------------+ + | Authorized | w/ Access Token + +--------------+ + | + | Fetch user info using the access token + v + +--------+ + | Done | + +--------+ + +(\*) The 'Errored' state hasn't been represented here for simplicity. + +-} +type Flow + = Idle + | Authorized OAuth.Token + | Done UserInfo + | Errored Error + + +type Error + = ErrStateMismatch + | ErrAuthorization OAuth.AuthorizationError + | ErrHTTPGetUserInfo + + +type alias UserInfo = + { name : String + , picture : String + } + + +type alias Configuration = + { authorizationEndpoint : Url + , userInfoEndpoint : Url + , userInfoDecoder : Json.Decoder UserInfo + , clientId : String + , scope : List String + } + + +{-| During the authentication flow, we'll run twice into the `init` function: + + - The first time, for the application very first run. And we proceed with the `Idle` state, + waiting for the user (a.k.a you) to request a sign in. + + - The second time, after a sign in has been requested, the user is redirected to the + authorization server and redirects the user back to our application, with an access + token and other fields as query parameters. + +When query params are present (and valid), we consider the user `Authorized`. + +-} +init : Maybe { state : String } -> Url -> Key -> ( Model, Cmd Msg ) +init mflags origin navigationKey = + let + redirectUri = + { origin | query = Nothing, fragment = Nothing } + + clearUrl = + Navigation.replaceUrl navigationKey (Url.toString redirectUri) + in + case OAuth.parseToken origin of + OAuth.Empty -> + ( { flow = Idle, redirectUri = redirectUri } + , Cmd.none + ) + + -- It is important to set a `state` when making the authorization request + -- and to verify it after the redirection. The state can be anything but its primary + -- usage is to prevent cross-site request forgery; at minima, it should be a short, + -- non-guessable string, generated on the fly. + -- + -- We remember any previously generated state state using the browser's local storage + -- and give it back (if present) to the elm application upon start + OAuth.Success { token, state } -> + case mflags of + Nothing -> + ( { flow = Errored ErrStateMismatch, redirectUri = redirectUri } + , clearUrl + ) + + Just flags -> + if state /= Just flags.state then + ( { flow = Errored ErrStateMismatch, redirectUri = redirectUri } + , clearUrl + ) + + else + ( { flow = Authorized token, redirectUri = redirectUri } + , Cmd.batch + -- Artificial delay to make the live demo easier to follow. + -- In practice, the access token could be requested right here. + [ after 750 Millisecond UserInfoRequested + , clearUrl + ] + ) + + OAuth.Error error -> + ( { flow = Errored <| ErrAuthorization error, redirectUri = redirectUri } + , clearUrl + ) + + + +-- +-- Msg +-- + + +type Msg + = NoOp + | SignInRequested + | GotRandomBytes (List Int) + | GotAccessToken (Result Http.Error OAuth.AuthorizationSuccess) + | UserInfoRequested + | GotUserInfo (Result Http.Error UserInfo) + | SignOutRequested + + +getUserInfo : Configuration -> OAuth.Token -> Cmd Msg +getUserInfo { userInfoDecoder, userInfoEndpoint } token = + Http.request + { method = "GET" + , body = Http.emptyBody + , headers = OAuth.useToken token [] + , url = Url.toString userInfoEndpoint + , expect = Http.expectJson GotUserInfo userInfoDecoder + , timeout = Nothing + , tracker = Nothing + } + + + +{- On the JavaScript's side, we have: + + app.ports.genRandomBytes.subscribe(n => { + const buffer = new Uint8Array(n); + crypto.getRandomValues(buffer); + const bytes = Array.from(buffer); + localStorage.setItem("bytes", bytes); + app.ports.randomBytes.send(bytes); + }); +-} + + +port genRandomBytes : Int -> Cmd msg + + +port randomBytes : (List Int -> msg) -> Sub msg + + + +-- +-- Update +-- + + +update : Msg -> Model -> ( Model, Cmd Msg ) +update msg model = + case ( model.flow, msg ) of + ( Idle, SignInRequested ) -> + signInRequested model + + ( Idle, GotRandomBytes bytes ) -> + gotRandomBytes model bytes + + ( Authorized token, UserInfoRequested ) -> + userInfoRequested model token + + ( Authorized _, GotUserInfo userInfoResponse ) -> + gotUserInfo model userInfoResponse + + ( Done _, SignOutRequested ) -> + signOutRequested model + + _ -> + noOp model + + +noOp : Model -> ( Model, Cmd Msg ) +noOp model = + ( model, Cmd.none ) + + +signInRequested : Model -> ( Model, Cmd Msg ) +signInRequested model = + ( { model | flow = Idle } + , genRandomBytes 16 + ) + + +gotRandomBytes : Model -> List Int -> ( Model, Cmd Msg ) +gotRandomBytes model bytes = + let + { state } = + convertBytes bytes + + authorization = + { clientId = configuration.clientId + , redirectUri = model.redirectUri + , scope = configuration.scope + , state = Just state + , url = configuration.authorizationEndpoint + } + in + ( { model | flow = Idle } + , authorization + |> OAuth.makeAuthorizationUrl + |> Url.toString + |> Navigation.load + ) + + +userInfoRequested : Model -> OAuth.Token -> ( Model, Cmd Msg ) +userInfoRequested model token = + ( { model | flow = Authorized token } + , getUserInfo configuration token + ) + + +gotUserInfo : Model -> Result Http.Error UserInfo -> ( Model, Cmd Msg ) +gotUserInfo model userInfoResponse = + case userInfoResponse of + Err _ -> + ( { model | flow = Errored ErrHTTPGetUserInfo } + , Cmd.none + ) + + Ok userInfo -> + ( { model | flow = Done userInfo } + , Cmd.none + ) + + +signOutRequested : Model -> ( Model, Cmd Msg ) +signOutRequested model = + ( { model | flow = Idle } + , Navigation.load (Url.toString model.redirectUri) + ) + + + +-- +-- View +-- + + +type alias ViewConfiguration msg = + { title : String + , btnClass : Attribute msg + } + + +view : ViewConfiguration Msg -> Model -> Document Msg +view ({ title } as config) model = + { title = title + , body = viewBody config model + } + + +viewBody : ViewConfiguration Msg -> Model -> List (Html Msg) +viewBody config model = + [ div [ class "flex", class "flex-column", class "flex-space-around" ] <| + case model.flow of + Idle -> + div [ class "flex" ] + [ viewAuthorizationStep False + , viewStepSeparator False + , viewGetUserInfoStep False + ] + :: viewIdle config + + Authorized _ -> + div [ class "flex" ] + [ viewAuthorizationStep True + , viewStepSeparator True + , viewGetUserInfoStep False + ] + :: viewAuthorized + + Done userInfo -> + div [ class "flex" ] + [ viewAuthorizationStep True + , viewStepSeparator True + , viewGetUserInfoStep True + ] + :: viewUserInfo config userInfo + + Errored err -> + div [ class "flex" ] + [ viewErroredStep + ] + :: viewErrored err + ] + + +viewIdle : ViewConfiguration Msg -> List (Html Msg) +viewIdle { btnClass } = + [ button + [ onClick SignInRequested, btnClass ] + [ text "Sign in" ] + ] + + +viewAuthorized : List (Html Msg) +viewAuthorized = + [ span [] [ text "Getting user info..." ] + ] + + +viewUserInfo : ViewConfiguration Msg -> UserInfo -> List (Html Msg) +viewUserInfo { btnClass } { name, picture } = + [ div [ class "flex", class "flex-column" ] + [ img [ class "avatar", src picture ] [] + , p [] [ text name ] + , div [] + [ button + [ onClick SignOutRequested, btnClass ] + [ text "Sign out" ] + ] + ] + ] + + +viewErrored : Error -> List (Html Msg) +viewErrored error = + [ span [ class "span-error" ] [ viewError error ] ] + + +viewError : Error -> Html Msg +viewError e = + text <| + case e of + ErrStateMismatch -> + "'state' doesn't match, the request has likely been forged by an adversary!" + + ErrAuthorization error -> + oauthErrorToString { error = error.error, errorDescription = error.errorDescription } + + ErrHTTPGetUserInfo -> + "Unable to retrieve user info: HTTP request failed." + + +viewAuthorizationStep : Bool -> Html Msg +viewAuthorizationStep isActive = + viewStep isActive ( "Authorization", style "left" "-110%" ) + + +viewGetUserInfoStep : Bool -> Html Msg +viewGetUserInfoStep isActive = + viewStep isActive ( "Get User Info", style "left" "-135%" ) + + +viewErroredStep : Html Msg +viewErroredStep = + div + [ class "step", class "step-errored" ] + [ span [ style "left" "-50%" ] [ text "Errored" ] ] + + +viewStep : Bool -> ( String, Attribute Msg ) -> Html Msg +viewStep isActive ( step, position ) = + let + stepClass = + class "step" + :: (if isActive then + [ class "step-active" ] + + else + [] + ) + in + div stepClass [ span [ position ] [ text step ] ] + + +viewStepSeparator : Bool -> Html Msg +viewStepSeparator isActive = + let + stepClass = + class "step-separator" + :: (if isActive then + [ class "step-active" ] + + else + [] + ) + in + span stepClass [] + + + +-- +-- Helpers +-- + + +toBytes : List Int -> Bytes +toBytes = + List.map Bytes.unsignedInt8 >> Bytes.sequence >> Bytes.encode + + +base64 : Bytes -> String +base64 = + Base64.bytes >> Base64.encode + + +convertBytes : List Int -> { state : String } +convertBytes = + toBytes >> base64 >> (\state -> { state = state }) + + +oauthErrorToString : { error : OAuth.ErrorCode, errorDescription : Maybe String } -> String +oauthErrorToString { error, errorDescription } = + let + desc = + errorDescription |> Maybe.withDefault "" |> String.replace "+" " " + in + OAuth.errorCodeToString error ++ ": " ++ desc + + +defaultHttpsUrl : Url +defaultHttpsUrl = + { protocol = Https + , host = "" + , path = "" + , port_ = Nothing + , query = Nothing + , fragment = Nothing + } diff --git a/examples/providers/spotify/implicit/README.md b/examples/providers/spotify/implicit/README.md new file mode 100644 index 0000000..e2e07a7 --- /dev/null +++ b/examples/providers/spotify/implicit/README.md @@ -0,0 +1,5 @@ +# Auth0 + +

+ +

diff --git a/examples/providers/spotify/implicit/elm.json b/examples/providers/spotify/implicit/elm.json new file mode 100644 index 0000000..a933f27 --- /dev/null +++ b/examples/providers/spotify/implicit/elm.json @@ -0,0 +1,34 @@ +{ + "type": "application", + "source-directories": [ + "src", + "." + ], + "elm-version": "0.19.1", + "dependencies": { + "direct": { + "andrewMacmurray/elm-delay": "3.0.0", + "elm/browser": "1.0.2", + "elm/bytes": "1.0.8", + "elm/core": "1.0.4", + "elm/html": "1.0.0", + "elm/http": "2.0.0", + "elm/json": "1.1.3", + "elm/url": "1.0.0", + "folkertdev/elm-sha2": "1.0.0", + "ivadzy/bbase64": "1.1.1" + }, + "indirect": { + "danfishgold/base64-bytes": "1.0.3", + "elm/file": "1.0.5", + "elm/regex": "1.0.0", + "elm/time": "1.0.0", + "elm/virtual-dom": "1.0.2", + "rtfeldman/elm-hex": "1.0.0" + } + }, + "test-dependencies": { + "direct": {}, + "indirect": {} + } +} diff --git a/examples/providers/spotify/implicit/src b/examples/providers/spotify/implicit/src new file mode 120000 index 0000000..dd81945 --- /dev/null +++ b/examples/providers/spotify/implicit/src @@ -0,0 +1 @@ +../../../../src/ \ No newline at end of file diff --git a/examples/src b/examples/src deleted file mode 120000 index 5cd551c..0000000 --- a/examples/src +++ /dev/null @@ -1 +0,0 @@ -../src \ No newline at end of file diff --git a/guides/facebook/README.md b/guides/facebook/README.md deleted file mode 100644 index 107d6ce..0000000 --- a/guides/facebook/README.md +++ /dev/null @@ -1,62 +0,0 @@ -# Interacting with Facebook - -Facebook has several quirks in its implementation. Therefore, in order to implement the _Implicit Flow_ -correctly, one needs to provide custom parsers to accomodate with Facebook specific format. Step-by-step: - -No 'token\_type' is returned, so we have to provide a default one as `Just "Bearer"` - -```elm -tokenParser : Query.Parser (Maybe OAuth.Token) -tokenParser = - Query.map (OAuth.makeToken (Just "Bearer")) - (Query.string "access_token") -``` - -In case of error, no 'error' field is returned, but instead we find a field named 'error\_code' - -```elm -errorParser : Query.Parser (Maybe OAuth.ErrorCode) -errorParser = - Query.map (Maybe.map OAuth.errorCodeFromString) - (Query.string "error_code") -``` - -Similarly, no 'error_description' is part of the error response, but instead we find an 'error\_message': - -```elm -authorizationErrorParser : OAuth.ErrorCode -> Query.Parser OAuth.Implicit.AuthorizationError -authorizationErrorParser errorCode = - Query.map3 (OAuth.Implicit.AuthorizationError errorCode) - (Query.string "error_message") - (Query.string "error_uri") - (Query.string "state") -``` - -In addition, parameters are returned as query parameters instead of a fragments, and _sometimes_, a noise fragment is present in the response. -So, as a work-around, one can patch the `Url` to make it compliant with the original RFC specification as follows: - -```elm -patchUrl : Url -> Url -patchUrl url = - if url.fragment == Just "_=_" || url.fragment == Nothing then - { url | fragment = url.query } - - _ -> - url -``` - -All-in-all, the `OAuth.Implicit.parseTokenWith` function can be used to put everything together -and parse redirect OAuth urls from Facebook authorization server: - -```elm -let parsers = - { tokenParser = tokenParser - , errorParser = errorParser - , authorizationSuccessParser = OAuth.Implicit.defaultAuthorizationSuccessParser - , authorizationErrorParser = authorizationErrorParser - } - -parseTokenWith parsers (patchUrl url) == _ : AuthorizationResult -``` - -:tada: diff --git a/guides/facebook/logo.png b/guides/facebook/logo.png deleted file mode 100644 index 32a0bdc42d87856a2ee8c75ecc1af058657d4c7c..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 11751 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 xe-~3PJN^FwVEcc8;{RR1_8+RegVX Http.Request AuthenticationSuccess -adjustRequest req = - let - headers = - Http.header "Accept" ("application/json") :: req.headers - - expect = - Http.expectJson AuthenticationSuccess <| Json.map4 - defaultTokenDecoder - defaultRefreshTokenDecoder - defaultExpiresInDecoder - lenientScopeDecoder - in - { req | headers = headers, expect = expect } - - -getToken : String -> Cmd AuthenticationSuccess -getToken code = - let - req = - adjustRequest <| - OAuth.AuthorizationCode.makeTokenRequest AuthenticationSuccess <| - OAuth.AuthorizationCode - { credentials = { clientId = clientId, secret = Nothing } - , code = code - , redirectUri = redirectUri - , scope = scope - , state = state - , url = tokenEndpoint - } - in - Http.send handleResponse (Http.request req) -``` diff --git a/guides/github/logo.png b/guides/github/logo.png deleted file mode 100644 index aa22ce244463582ff8aef9e2966f06c120ec42b8..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 9924 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