mirror of
https://github.com/Dvorinka/excalidraw-full.git
synced 2026-06-03 13:52:56 +00:00
179 lines
4.7 KiB
Go
179 lines
4.7 KiB
Go
package auth
|
|
|
|
import (
|
|
"context"
|
|
"crypto/rand"
|
|
"encoding/base64"
|
|
"encoding/json"
|
|
"excalidraw-complete/core"
|
|
"fmt"
|
|
"io"
|
|
"net/http"
|
|
"os"
|
|
"time"
|
|
|
|
"github.com/golang-jwt/jwt/v5"
|
|
"github.com/sirupsen/logrus"
|
|
"golang.org/x/oauth2"
|
|
"golang.org/x/oauth2/github"
|
|
)
|
|
|
|
var (
|
|
githubOauthConfig *oauth2.Config
|
|
jwtSecret []byte
|
|
)
|
|
|
|
|
|
// AppClaims represents the custom claims for the JWT.
|
|
type AppClaims struct {
|
|
jwt.RegisteredClaims
|
|
Login string `json:"login"`
|
|
AvatarURL string `json:"avatarUrl"`
|
|
Name string `json:"name"`
|
|
}
|
|
|
|
func Init() {
|
|
githubOauthConfig = &oauth2.Config{
|
|
ClientID: os.Getenv("GITHUB_CLIENT_ID"),
|
|
ClientSecret: os.Getenv("GITHUB_CLIENT_SECRET"),
|
|
RedirectURL: os.Getenv("GITHUB_REDIRECT_URL"),
|
|
Scopes: []string{"read:user", "user:email"},
|
|
Endpoint: github.Endpoint,
|
|
}
|
|
jwtSecret = []byte(os.Getenv("JWT_SECRET"))
|
|
|
|
if githubOauthConfig.ClientID == "" || githubOauthConfig.ClientSecret == "" {
|
|
logrus.Warn("GitHub OAuth credentials are not set. Authentication routes will not work.")
|
|
}
|
|
if len(jwtSecret) == 0 {
|
|
logrus.Warn("JWT_SECRET is not set. Authentication routes will not work.")
|
|
}
|
|
}
|
|
|
|
func generateStateOauthCookie(w http.ResponseWriter) string {
|
|
b := make([]byte, 16)
|
|
rand.Read(b)
|
|
state := base64.URLEncoding.EncodeToString(b)
|
|
cookie := &http.Cookie{
|
|
Name: "oauthstate",
|
|
Value: state,
|
|
Expires: time.Now().Add(10 * time.Minute),
|
|
HttpOnly: true,
|
|
}
|
|
http.SetCookie(w, cookie)
|
|
return state
|
|
}
|
|
|
|
func HandleGitHubLogin(w http.ResponseWriter, r *http.Request) {
|
|
if githubOauthConfig.ClientID == "" {
|
|
http.Error(w, "GitHub OAuth is not configured", http.StatusInternalServerError)
|
|
return
|
|
}
|
|
state := generateStateOauthCookie(w)
|
|
url := githubOauthConfig.AuthCodeURL(state)
|
|
http.Redirect(w, r, url, http.StatusTemporaryRedirect)
|
|
}
|
|
|
|
func HandleGitHubCallback(w http.ResponseWriter, r *http.Request) {
|
|
if githubOauthConfig.ClientID == "" {
|
|
http.Error(w, "GitHub OAuth is not configured", http.StatusInternalServerError)
|
|
return
|
|
}
|
|
|
|
oauthState, _ := r.Cookie("oauthstate")
|
|
if r.FormValue("state") != oauthState.Value {
|
|
logrus.Error("invalid oauth github state")
|
|
http.Redirect(w, r, "/", http.StatusTemporaryRedirect)
|
|
return
|
|
}
|
|
|
|
token, err := githubOauthConfig.Exchange(context.Background(), r.FormValue("code"))
|
|
if err != nil {
|
|
logrus.Errorf("failed to exchange token: %s", err.Error())
|
|
http.Redirect(w, r, "/", http.StatusTemporaryRedirect)
|
|
return
|
|
}
|
|
|
|
client := githubOauthConfig.Client(context.Background(), token)
|
|
resp, err := client.Get("https://api.github.com/user")
|
|
if err != nil {
|
|
logrus.Errorf("failed to get user from github: %s", err.Error())
|
|
http.Redirect(w, r, "/", http.StatusTemporaryRedirect)
|
|
return
|
|
}
|
|
defer resp.Body.Close()
|
|
|
|
body, err := io.ReadAll(resp.Body)
|
|
if err != nil {
|
|
logrus.Errorf("failed to read github response body: %s", err.Error())
|
|
http.Redirect(w, r, "/", http.StatusTemporaryRedirect)
|
|
return
|
|
}
|
|
|
|
var githubUser struct {
|
|
ID int64 `json:"id"`
|
|
Login string `json:"login"`
|
|
AvatarURL string `json:"avatar_url"`
|
|
Name string `json:"name"`
|
|
}
|
|
|
|
if err := json.Unmarshal(body, &githubUser); err != nil {
|
|
logrus.Errorf("failed to unmarshal github user: %s", err.Error())
|
|
http.Redirect(w, r, "/", http.StatusTemporaryRedirect)
|
|
return
|
|
}
|
|
|
|
// Create user object using Subject instead of GitHubID
|
|
user := &core.User{
|
|
Subject: fmt.Sprintf("github:%d", githubUser.ID),
|
|
Login: githubUser.Login,
|
|
AvatarURL: githubUser.AvatarURL,
|
|
Name: githubUser.Name,
|
|
}
|
|
|
|
jwtToken, err := createJWT(user)
|
|
if err != nil {
|
|
logrus.Errorf("failed to create JWT: %s", err.Error())
|
|
http.Redirect(w, r, "/", http.StatusTemporaryRedirect)
|
|
return
|
|
}
|
|
|
|
// Redirect to frontend with token
|
|
http.Redirect(w, r, fmt.Sprintf("/?token=%s", jwtToken), http.StatusTemporaryRedirect)
|
|
}
|
|
|
|
func createJWT(user *core.User) (string, error) {
|
|
claims := AppClaims{
|
|
RegisteredClaims: jwt.RegisteredClaims{
|
|
Subject: user.Subject,
|
|
ExpiresAt: jwt.NewNumericDate(time.Now().Add(time.Hour * 24 * 7)), // 1 week
|
|
IssuedAt: jwt.NewNumericDate(time.Now()),
|
|
},
|
|
Login: user.Login,
|
|
AvatarURL: user.AvatarURL,
|
|
Name: user.Name,
|
|
}
|
|
|
|
token := jwt.NewWithClaims(jwt.SigningMethodHS256, claims)
|
|
return token.SignedString(jwtSecret)
|
|
}
|
|
|
|
func ParseJWT(tokenString string) (*AppClaims, error) {
|
|
token, err := jwt.ParseWithClaims(tokenString, &AppClaims{}, func(token *jwt.Token) (interface{}, error) {
|
|
if _, ok := token.Method.(*jwt.SigningMethodHMAC); !ok {
|
|
return nil, fmt.Errorf("unexpected signing method: %v", token.Header["alg"])
|
|
}
|
|
return jwtSecret, nil
|
|
})
|
|
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if claims, ok := token.Claims.(*AppClaims); ok && token.Valid {
|
|
return claims, nil
|
|
}
|
|
|
|
return nil, fmt.Errorf("invalid token")
|
|
}
|