All checks were successful
Deploy with Docker Compose / deploy (push) Successful in 4m44s
- Fix auth bypass in UpdatePost/DeletePost (missing return after auth check) - Remove Spotify access token from callback response - Replace internal error messages with generic responses in all handlers - Harden GraphQL: complexity limit, disable playground/introspection in prod - Add security headers (X-Frame-Options, HSTS, etc.) to nginx - Disable Hasura console/dev mode in production - Add DOMPurify sanitization to Markdown component - Fix cookie removal to use correct domain/path from auth config - Fix nil dereference in rowing handler when Claude API errors - Fix wildcard CORS on stamp endpoint - Pin nginx and certbot Docker image versions Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
213 lines
4.7 KiB
Go
213 lines
4.7 KiB
Go
package handlers
|
|
|
|
import (
|
|
"log"
|
|
"net/http"
|
|
|
|
"adam-french.co.uk/backend/models"
|
|
"github.com/gin-gonic/gin"
|
|
"github.com/golang-jwt/jwt/v5"
|
|
"golang.org/x/crypto/bcrypt"
|
|
)
|
|
|
|
func (store *Store) AuthMiddlewear(ctx *gin.Context) {
|
|
access_token, err := ctx.Cookie("access_token")
|
|
if err != nil {
|
|
ctx.AbortWithStatusJSON(401, gin.H{"error": "unauthorized"})
|
|
return
|
|
}
|
|
|
|
claims, err := store.Auth.VerifyJWT(access_token)
|
|
if err != nil {
|
|
ctx.AbortWithStatusJSON(401, gin.H{"error": "unauthorized"})
|
|
return
|
|
}
|
|
|
|
// store claims in Gin context
|
|
ctx.Set("userClaims", claims)
|
|
ctx.Next()
|
|
}
|
|
|
|
func (store *Store) AdminMiddleware(ctx *gin.Context) {
|
|
claims, exists := ctx.Get("userClaims")
|
|
if !exists {
|
|
ctx.AbortWithStatusJSON(http.StatusUnauthorized, gin.H{"error": "unauthorized"})
|
|
return
|
|
}
|
|
|
|
mapClaims, ok := claims.(*jwt.MapClaims)
|
|
if !ok {
|
|
ctx.AbortWithStatusJSON(http.StatusUnauthorized, gin.H{"error": "invalid claims"})
|
|
return
|
|
}
|
|
|
|
admin, ok := (*mapClaims)["admin"].(bool)
|
|
if !ok || !admin {
|
|
ctx.AbortWithStatusJSON(http.StatusForbidden, gin.H{"error": "admin access required"})
|
|
return
|
|
}
|
|
|
|
ctx.Next()
|
|
}
|
|
|
|
func (store *Store) CheckToken(ctx *gin.Context) {
|
|
access_token, err := ctx.Cookie("access_token")
|
|
if err != nil {
|
|
ctx.JSON(401, gin.H{"error": "unauthorized"})
|
|
return
|
|
}
|
|
|
|
claims, err := store.Auth.VerifyJWT(access_token)
|
|
if err != nil {
|
|
ctx.JSON(401, gin.H{"error": "unauthorized"})
|
|
return
|
|
}
|
|
|
|
userIDF, ok := (*claims)["id"].(float64)
|
|
if !ok {
|
|
ctx.JSON(401, gin.H{"error": "unauthorized"})
|
|
return
|
|
}
|
|
userID := uint(userIDF)
|
|
|
|
user := models.User{ID: userID}
|
|
tx := store.DB.First(&user)
|
|
if tx.Error != nil {
|
|
log.Println(tx.Error)
|
|
ctx.JSON(http.StatusNotFound, gin.H{"error": "not found"})
|
|
store.removeCookies(ctx)
|
|
return
|
|
}
|
|
|
|
ctx.JSON(http.StatusOK, user)
|
|
}
|
|
|
|
func (store *Store) RefreshToken(ctx *gin.Context) {
|
|
refreshToken, err := ctx.Cookie("refresh_token")
|
|
if err != nil {
|
|
ctx.JSON(http.StatusUnauthorized, gin.H{"error": "unauthorized"})
|
|
return
|
|
}
|
|
|
|
claims, err := store.Auth.VerifyJWT(refreshToken)
|
|
if err != nil {
|
|
ctx.JSON(http.StatusUnauthorized, gin.H{"error": "unauthorized"})
|
|
return
|
|
}
|
|
|
|
userIDF, ok := (*claims)["id"].(float64)
|
|
if !ok {
|
|
ctx.JSON(http.StatusInternalServerError, gin.H{"error": "invalid token claims"})
|
|
return
|
|
}
|
|
userID := uint(userIDF)
|
|
|
|
user := models.User{ID: userID}
|
|
tx := store.DB.First(&user)
|
|
if tx.Error != nil {
|
|
log.Println(tx.Error)
|
|
ctx.JSON(http.StatusNotFound, gin.H{"error": "not found"})
|
|
store.removeCookies(ctx)
|
|
return
|
|
}
|
|
|
|
tokens, err := store.Auth.GenerateJWT(&user)
|
|
if err != nil {
|
|
log.Println(err)
|
|
ctx.JSON(http.StatusInternalServerError, gin.H{"error": "internal error"})
|
|
return
|
|
}
|
|
|
|
ctx.SetSameSite(http.SameSiteLaxMode)
|
|
ctx.SetCookie(
|
|
"access_token",
|
|
tokens.AccessToken,
|
|
int(store.Auth.Config.AccessTokenLifetime.Seconds()),
|
|
store.Auth.Config.Endpoint,
|
|
store.Auth.Config.Domain,
|
|
true, true,
|
|
)
|
|
ctx.SetCookie(
|
|
"refresh_token",
|
|
tokens.RefreshToken,
|
|
int(store.Auth.Config.RefreshTokenLifetime.Seconds()),
|
|
store.Auth.Config.Endpoint,
|
|
store.Auth.Config.Domain,
|
|
true, true,
|
|
)
|
|
|
|
ctx.JSON(http.StatusAccepted, user)
|
|
}
|
|
|
|
func (store *Store) Login(ctx *gin.Context) {
|
|
var input UserCredentials
|
|
if err := ctx.ShouldBindBodyWithJSON(&input); err != nil {
|
|
ctx.JSON(http.StatusBadRequest, gin.H{"error": "invalid request body"})
|
|
return
|
|
}
|
|
|
|
user := models.User{}
|
|
if err := store.DB.Where("username = ?", input.Username).First(&user).Error; err != nil {
|
|
ctx.JSON(http.StatusUnauthorized, gin.H{"error": "invalid credentials"})
|
|
return
|
|
}
|
|
|
|
if err := bcrypt.CompareHashAndPassword(user.Password, []byte(input.Password)); err != nil {
|
|
ctx.JSON(http.StatusUnauthorized, gin.H{"error": "invalid credentials"})
|
|
return
|
|
}
|
|
|
|
tokens, err := store.Auth.GenerateJWT(&user)
|
|
if err != nil {
|
|
log.Println(err)
|
|
ctx.JSON(http.StatusInternalServerError, gin.H{"error": "internal error"})
|
|
return
|
|
}
|
|
|
|
ctx.SetSameSite(http.SameSiteLaxMode)
|
|
ctx.SetCookie(
|
|
"access_token",
|
|
tokens.AccessToken,
|
|
int(store.Auth.Config.AccessTokenLifetime.Seconds()),
|
|
store.Auth.Config.Endpoint,
|
|
store.Auth.Config.Domain,
|
|
true, true,
|
|
)
|
|
ctx.SetCookie(
|
|
"refresh_token",
|
|
tokens.RefreshToken,
|
|
int(store.Auth.Config.RefreshTokenLifetime.Seconds()),
|
|
store.Auth.Config.Endpoint,
|
|
store.Auth.Config.Domain,
|
|
true, true,
|
|
)
|
|
|
|
ctx.JSON(http.StatusAccepted, user)
|
|
}
|
|
|
|
func (store *Store) Logout(ctx *gin.Context) {
|
|
store.removeCookies(ctx)
|
|
|
|
ctx.Status(http.StatusOK)
|
|
}
|
|
|
|
func (store *Store) removeCookies(ctx *gin.Context) {
|
|
ctx.SetSameSite(http.SameSiteLaxMode)
|
|
ctx.SetCookie(
|
|
"access_token",
|
|
"",
|
|
-1,
|
|
store.Auth.Config.Endpoint,
|
|
store.Auth.Config.Domain,
|
|
true, true,
|
|
)
|
|
ctx.SetCookie(
|
|
"refresh_token",
|
|
"",
|
|
-1,
|
|
store.Auth.Config.Endpoint,
|
|
store.Auth.Config.Domain,
|
|
true, true,
|
|
)
|
|
}
|