177 lines
3.7 KiB
Go
177 lines
3.7 KiB
Go
package openid
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"errors"
|
|
"fmt"
|
|
"net/http"
|
|
"time"
|
|
|
|
"github.com/coreos/go-oidc"
|
|
"github.com/google/uuid"
|
|
"github.com/marcopeocchi/yt-dlp-web-ui/server/config"
|
|
"golang.org/x/oauth2"
|
|
)
|
|
|
|
type OAuth2SuccessResponse struct {
|
|
OAuth2Token *oauth2.Token
|
|
OAuth2RawToken string
|
|
IDTokenClaims *json.RawMessage
|
|
}
|
|
|
|
var (
|
|
oauth2Config oauth2.Config
|
|
verifier *oidc.IDTokenVerifier
|
|
)
|
|
|
|
func Configure() {
|
|
if !config.Instance().UseOpenId {
|
|
return
|
|
}
|
|
|
|
provider, err := oidc.NewProvider(context.Background(), config.Instance().OpenIdProviderURL)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
oauth2Config = oauth2.Config{
|
|
ClientID: config.Instance().OpenIdClientId,
|
|
ClientSecret: config.Instance().OpenIdClientSecret,
|
|
RedirectURL: config.Instance().OpenIdRedirectURL,
|
|
Endpoint: provider.Endpoint(),
|
|
Scopes: []string{oidc.ScopeOpenID, "profile", "email"},
|
|
}
|
|
|
|
verifier = provider.Verifier(&oidc.Config{
|
|
ClientID: config.Instance().OpenIdClientId,
|
|
})
|
|
}
|
|
|
|
func Login(w http.ResponseWriter, r *http.Request) {
|
|
var (
|
|
state = uuid.NewString()
|
|
nonce = uuid.NewString() // maybe something cryptographycally more seucre?
|
|
)
|
|
|
|
http.SetCookie(w, &http.Cookie{
|
|
Name: "state",
|
|
Value: state,
|
|
HttpOnly: true,
|
|
Path: "/",
|
|
Secure: r.TLS != nil,
|
|
Expires: time.Now().Add(time.Hour * 24 * 30),
|
|
})
|
|
|
|
http.SetCookie(w, &http.Cookie{
|
|
Name: "nonce",
|
|
Value: nonce,
|
|
HttpOnly: true,
|
|
Path: "/",
|
|
Secure: r.TLS != nil,
|
|
Expires: time.Now().Add(time.Hour * 24 * 30),
|
|
})
|
|
|
|
http.Redirect(w, r, oauth2Config.AuthCodeURL(state, oidc.Nonce(nonce)), http.StatusFound)
|
|
}
|
|
|
|
func doAuthentification(r *http.Request) (*OAuth2SuccessResponse, error) {
|
|
state, err := r.Cookie("state")
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if r.URL.Query().Get("state") != state.Value {
|
|
return nil, errors.New("auth state does not match")
|
|
}
|
|
|
|
oauth2Token, err := oauth2Config.Exchange(r.Context(), r.URL.Query().Get("code"))
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
rawToken, ok := oauth2Token.Extra("id_token").(string)
|
|
if !ok {
|
|
return nil, errors.New("openid field \"id_token\" not found in oauth2 token")
|
|
}
|
|
|
|
idToken, err := verifier.Verify(r.Context(), rawToken)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
nonce, err := r.Cookie("nonce")
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if idToken.Nonce != nonce.Value {
|
|
return nil, errors.New("auth nonce does not match")
|
|
}
|
|
|
|
// redact
|
|
oauth2Token.AccessToken = ""
|
|
|
|
res := OAuth2SuccessResponse{
|
|
oauth2Token,
|
|
rawToken,
|
|
&json.RawMessage{},
|
|
}
|
|
|
|
if err := idToken.Claims(&res.IDTokenClaims); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return &res, nil
|
|
}
|
|
|
|
func SingIn(w http.ResponseWriter, r *http.Request) {
|
|
res, err := doAuthentification(r)
|
|
if err != nil {
|
|
http.Error(w, err.Error(), http.StatusBadRequest)
|
|
return
|
|
}
|
|
|
|
http.SetCookie(w, &http.Cookie{
|
|
Name: "oid-token",
|
|
Value: res.OAuth2RawToken,
|
|
HttpOnly: true,
|
|
Path: "/",
|
|
Secure: r.TLS != nil,
|
|
Expires: time.Now().Add(time.Hour * 24 * 30),
|
|
})
|
|
|
|
// if err := json.NewEncoder(w).Encode(res); err != nil {
|
|
// http.Error(w, err.Error(), http.StatusInternalServerError)
|
|
// return
|
|
// }
|
|
|
|
fmt.Fprintf(w, "Login succesfully, you may now close this window and refresh yt-dlp-webui.")
|
|
}
|
|
|
|
func Logout(w http.ResponseWriter, r *http.Request) {
|
|
http.SetCookie(w, &http.Cookie{
|
|
Name: "oid-token",
|
|
HttpOnly: true,
|
|
Path: "/",
|
|
Secure: r.TLS != nil,
|
|
Expires: time.Now(),
|
|
})
|
|
|
|
http.SetCookie(w, &http.Cookie{
|
|
Name: "state",
|
|
HttpOnly: true,
|
|
Path: "/",
|
|
Secure: r.TLS != nil,
|
|
Expires: time.Now(),
|
|
})
|
|
|
|
http.SetCookie(w, &http.Cookie{
|
|
Name: "nonce",
|
|
HttpOnly: true,
|
|
Path: "/",
|
|
Secure: r.TLS != nil,
|
|
Expires: time.Now(),
|
|
})
|
|
}
|