Merge pull request 'Add authmw for other services to use as middleware' (#1) from authmw into master
Reviewed-on: #1
This commit is contained in:
commit
abbc102060
92
authmw/token.go
Normal file
92
authmw/token.go
Normal file
@ -0,0 +1,92 @@
|
||||
package authmw
|
||||
|
||||
import (
|
||||
"context"
|
||||
"crypto/x509"
|
||||
"encoding/json"
|
||||
"fmt"
|
||||
"net/http"
|
||||
"slices"
|
||||
"strings"
|
||||
|
||||
"git.t-juice.club/microfilm/auth"
|
||||
"github.com/golang-jwt/jwt/v5"
|
||||
)
|
||||
|
||||
func VerifyToken(authURL string, permittedRoles []string) func(http.Handler) http.Handler {
|
||||
// Fetch current pubkey
|
||||
url := fmt.Sprintf("%s/key", authURL)
|
||||
req, err := http.NewRequest(http.MethodGet, url, nil)
|
||||
if err != nil {
|
||||
panic(err)
|
||||
}
|
||||
|
||||
resp, err := http.DefaultClient.Do(req)
|
||||
if err != nil {
|
||||
panic(err)
|
||||
}
|
||||
defer resp.Body.Close()
|
||||
|
||||
var authResponse auth.PubkeyResponse
|
||||
decoder := json.NewDecoder(resp.Body)
|
||||
if err := decoder.Decode(&authResponse); err != nil {
|
||||
panic(err)
|
||||
}
|
||||
|
||||
// Parse pubkey
|
||||
pub, err := x509.ParsePKIXPublicKey(authResponse.PubKey)
|
||||
if err != nil {
|
||||
panic(err)
|
||||
}
|
||||
|
||||
fn := func(next http.Handler) http.Handler {
|
||||
fn := func(w http.ResponseWriter, r *http.Request) {
|
||||
authHeader := r.Header.Get("Authorization")
|
||||
if !strings.Contains(authHeader, "Bearer ") {
|
||||
// No token, pass if unathorized in permitted
|
||||
// else reject
|
||||
if slices.Contains[[]string, string](permittedRoles, auth.RoleUnauthorized) {
|
||||
next.ServeHTTP(w, r)
|
||||
return
|
||||
}
|
||||
|
||||
// Reject and write error response
|
||||
w.WriteHeader(http.StatusUnauthorized)
|
||||
var errResp auth.ErrorResponse
|
||||
errResp.Message = fmt.Sprintf("Authorization required: %s", strings.Join(permittedRoles, ","))
|
||||
errResp.Status = http.StatusUnauthorized
|
||||
|
||||
encoder := json.NewEncoder(w)
|
||||
_ = encoder.Encode(&errResp)
|
||||
return
|
||||
}
|
||||
|
||||
// Validate token
|
||||
tokenString := strings.Split(authHeader, " ")[1]
|
||||
token, err := jwt.ParseWithClaims(tokenString, &auth.MicrofilmClaims{}, func(t *jwt.Token) (interface{}, error) { return pub, nil })
|
||||
if err != nil {
|
||||
// Reject and write error response
|
||||
w.WriteHeader(http.StatusUnauthorized)
|
||||
var errResp auth.ErrorResponse
|
||||
errResp.Message = fmt.Sprintf("Token verification failed: %s", err)
|
||||
errResp.Status = http.StatusUnauthorized
|
||||
|
||||
encoder := json.NewEncoder(w)
|
||||
_ = encoder.Encode(&errResp)
|
||||
return
|
||||
}
|
||||
|
||||
// Add claims to request context
|
||||
if claims, ok := token.Claims.(*auth.MicrofilmClaims); ok && token.Valid {
|
||||
ctx := context.WithValue(r.Context(), "claims", claims)
|
||||
next.ServeHTTP(w, r.WithContext(ctx))
|
||||
return
|
||||
}
|
||||
|
||||
next.ServeHTTP(w, r)
|
||||
}
|
||||
return http.HandlerFunc(fn)
|
||||
}
|
||||
|
||||
return fn
|
||||
}
|
5
token.go
5
token.go
@ -3,8 +3,9 @@ package auth
|
||||
import "github.com/golang-jwt/jwt/v5"
|
||||
|
||||
const (
|
||||
RoleUser = "user"
|
||||
RoleAdmin = "admin"
|
||||
RoleUnauthorized = ""
|
||||
RoleUser = "user"
|
||||
RoleAdmin = "admin"
|
||||
)
|
||||
|
||||
type MicrofilmClaims struct {
|
||||
|
@ -1,3 +1,3 @@
|
||||
package auth
|
||||
|
||||
const Version = "v0.1.0"
|
||||
const Version = "v0.1.1"
|
||||
|
Loading…
Reference in New Issue
Block a user